advanced-portfolio-sorted (sort on any report column, such as gain and amount-out)

Johan van Oostrum jo.vanoost at wanadoo.nl
Sun Jan 21 06:13:08 EST 2007


Hi all,

The following hack may be of interest for those who quickly want an 
overview of their over/underperforming stocks. It is a slightly 
modified version of the advanced-portfolio report. Added is an option 
to choosea column the report should be sorted upon. I decided to name 
it advanced-portfolio-sorted to enable users to give it a try without 
removing the original report. Slight drawback of this approach is that 
one has to edit standard-reports.scm.

I would never have had the courage to start writing Scheme without the 
help of drScheme to balance parentheses. By the way, those of you 
developing with scheme should have a look at drScheme, if not already 
using this. http://www.drscheme.org/

Enfin, installation of the report (I use Fink and have the files 
located here: /sw/share/gnucash/guile-modules/gnucash/report/)
* 1 * Copy the report code to advance-portfolio-sorted.scm

* 2 * Add the marked line in standard-reports.scm:
...
(use-modules (gnucash report advanced-portfolio))
(use-modules (gnucash report advanced-portfolio-sorted)) ;; this is the 
new report
(use-modules (gnucash report average-balance))
...

* 3 * Following is the new report code (save as 
advanced-portfolio-sorted.scm):
;; -*-scheme-*- ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; advanced-portfolio-sorted.scm
;; by Johan van Oostrum (jvo at chaosgeordend.nl) Jan 2007
;;
;; This is a modified version of:
;; advanced-portfolio.scm
;; by Martijn van Oosterhout (kleptog at svana.org) Feb 2002,
;; modified for GnuCash 1.8 by Herbert Thoma (herbie at hthoma.de) Oct 2002
;; Which in turn is heavily based on portfolio.scm
;; by Robert Merkel (rgmerk at mira.net)
;;
;; This program is free software; you can redistribute it and/or
;; modify it under the terms of the GNU General Public License as
;; published by the Free Software Foundation; either version 2 of
;; the License, or (at your option) any later version.
;;
;; This program is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
;; GNU General Public License for more details.
;;
;; You should have received a copy of the GNU General Public License
;; along with this program; if not, contact:
;;
;; Free Software Foundation           Voice:  +1-617-542-5942
;; 59 Temple Place - Suite 330        Fax:    +1-617-542-2652
;; Boston, MA  02111-1307,  USA       gnu at gnu.org
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

(define-module (gnucash report advanced-portfolio-sorted))

(use-modules (gnucash main)) ;; FIXME: delete after we finish 
modularizing.
(use-modules (srfi srfi-1))
(use-modules (ice-9 slib))
(use-modules (gnucash gnc-module))

(require 'printf)

(gnc:module-load "gnucash/report/report-system" 0)

(define reportname (N_ "Advanced Portfolio Sorted"))

(define optname-price-source  (N_ "Price Source"))
(define optname-sort-column   (N_ "Sort Column"))
(define optname-shares-digits (N_ "Share decimal places"))
(define optname-zero-shares   (N_ "Include accounts with no shares"))
(define optname-include-gains (N_ "Include gains and losses"))

(define (options-generator)
   (let* ((options (gnc:new-options))
          ;; This is just a helper function for making options.
          ;; See gnucash/src/scm/options.scm for details.
          (add-option
           (lambda (new-option)
             (gnc:register-option options new-option))))

     ;; General Tab
     ;; date at which to report balance
     (gnc:options-add-report-date!
      options gnc:pagename-general
      (N_ "Date") "a")

     (gnc:options-add-currency!
      options gnc:pagename-general (N_ "Report Currency") "c")

     (add-option
      (gnc:make-multichoice-option
       gnc:pagename-general optname-sort-column
       "d" (N_ "The column on which the report is sorted") '0
       (list (vector '0 (N_ "Account") (N_ ""))
             (vector '1 (N_ "Symbol") (N_ ""))
             (vector '2 (N_ "Listing") (N_ ""))
             (vector '3 (N_ "Shares") (N_ ""))
             (vector '4 (N_ "Price") (N_ ""))
             (vector '5 (N_ "Value") (N_ ""))
             (vector '6 (N_ "Money-in") (N_ ""))
             (vector '7 (N_ "Money-out") (N_ ""))
             (vector '8 (N_ "Gain") (N_ ""))
             (vector '9 (N_ "Total return") (N_ ""))
             )))

     (add-option
      (gnc:make-multichoice-option
       gnc:pagename-general optname-price-source
       "d" (N_ "The source of price information") 'pricedb-nearest
       (list (vector 'pricedb-latest
                     (N_ "Most recent")
                     (N_ "The most recent recorded price"))
             (vector 'pricedb-nearest
                     (N_ "Nearest in time")
                     (N_ "The price recorded nearest in time to the 
report date"))
             )))

     (add-option
      (gnc:make-number-range-option
       gnc:pagename-general optname-shares-digits
       "e" (N_ "The number of decimal places to use for share numbers") 2
       0 6 0 1))

     (gnc:register-option
      options
      (gnc:make-simple-boolean-option
       gnc:pagename-general optname-include-gains "f"
       (N_ "Include splits with no shares for calculating money-in and 
money-out")
       #f))

     ;; Account tab
     (add-option
      (gnc:make-account-list-option
       gnc:pagename-accounts (N_ "Accounts")
       "b"
       (N_ "Stock Accounts to report on")
       (lambda () (filter gnc:account-is-stock?
                          (gnc:group-get-subaccounts
                           (gnc:get-current-group))))
       (lambda (accounts) (list  #t
                                 (filter gnc:account-is-stock? 
accounts)))
       #t))

     (gnc:register-option
      options
      (gnc:make-simple-boolean-option
       gnc:pagename-accounts optname-zero-shares "e"
       (N_ "Include accounts that have a zero share balances.")
       #f))

     (gnc:options-set-default-section options gnc:pagename-general)
     options))

;; This is the rendering function. It accepts a database of options
;; and generates an object of type <html-document>.  See the file
;; report-html.txt for documentation; the file report-html.scm
;; includes all the relevant Scheme code. The option database passed
;; to the function is one created by the options-generator function
;; defined above.
(define (advanced-portfolio-sorted-renderer report-obj)

   (let ((work-done 0)
         (work-to-do 0))

     ;; These are some helper functions for looking up option values.
     (define (get-op section name)
       (gnc:lookup-option (gnc:report-options report-obj) section name))

     (define (get-option section name)
       (gnc:option-value (get-op section name)))

     (define (split-account-type? split type)
       (eq? type
            (gw:enum-<gnc:AccountType>-val->sym (gnc:account-get-type 
(gnc:split-get-account split)) #f)))

     (define (same-split? s1 s2)
       (string=? (gnc:split-get-guid s1) (gnc:split-get-guid s2)))

     ;; return list with computed values for all selected stocks
     (define (table-add-stock-rows accounts to-date
                                   currency price-fn exchange-fn 
include-empty include-gains
                                   total-value total-moneyin 
total-moneyout total-gain)

       (let ()
         (define (table-add-stock-rows-internal accounts)
           (if (null? accounts) (list) ; return empty list
               (let* ((current (car accounts))
                      (rest (cdr accounts))

                      (name (gnc:account-get-name current))
                      (commodity (gnc:account-get-commodity current))
                      (ticker-symbol (gnc:commodity-get-mnemonic 
commodity))
                      (listing (gnc:commodity-get-namespace commodity))
                      (unit-collector 
(gnc:account-get-comm-balance-at-date
                                       current to-date #f))
                      (units (cadr (unit-collector 'getpair commodity 
#f)))
                      (totalunits 0.0)
                      (totalunityears 0.0)

                      ;; Counter to keep track of stuff
                      (unitscoll     (gnc:make-commodity-collector))
                      (brokeragecoll (gnc:make-commodity-collector))
                      (dividendcoll  (gnc:make-commodity-collector))
                      (moneyincoll   (gnc:make-commodity-collector))
                      (moneyoutcoll  (gnc:make-commodity-collector))
                      (gaincoll      (gnc:make-commodity-collector))

                      (price-list (price-fn commodity to-date))
                      (price      (if (> (length price-list) 0)
                                      (car price-list)
                                      #f))
                      (rate       (if price
                                      (gnc:make-gnc-monetary
                                       (gnc:price-get-currency price)
                                       (gnc:price-get-value price))
                                      #f))
                      (value (exchange-fn (gnc:make-gnc-monetary 
commodity units) currency to-date))
                      )

                 ;; (gnc:debug "---" name "---")
                 (for-each
                  (lambda (split)
                    (set! work-done (+ 1 work-done))
                    (gnc:report-percent-done (* 100 (/ work-done 
work-to-do)))
                    (let ((parent (gnc:split-get-parent split)))
                      (if (gnc:timepair-le 
(gnc:transaction-get-date-posted parent) to-date)
                          (for-each
                           (lambda (s)
                             (cond
                               ((same-split? s split)
                                ;; (gnc:debug "amount" 
(gnc:numeric-to-double (gnc:split-get-amount s)) )
                                (cond
                                  ((or include-gains (not 
(gnc:numeric-zero-p (gnc:split-get-amount s))))
                                   (unitscoll 'add commodity 
(gnc:split-get-amount s)) ;; Is the stock transaction?
                                   (if (< 0 (gnc:numeric-to-double
                                             (gnc:split-get-amount s)))
                                       (set! totalunits
                                             (+ totalunits
                                                (gnc:numeric-to-double 
(gnc:split-get-amount s)))))
                                   (set! totalunityears
                                         (+ totalunityears
                                            (* (gnc:numeric-to-double 
(gnc:split-get-amount s))
                                               (gnc:date-year-delta
                                                (car 
(gnc:transaction-get-date-posted parent))
                                                (current-time)))))
                                   (cond
                                     ((gnc:numeric-negative-p 
(gnc:split-get-value s))
                                      (moneyoutcoll
                                       'add currency
                                       (gnc:numeric-neg 
(gnc:split-get-value s))))
                                     (else (moneyincoll
                                            'add currency
                                            (gnc:numeric-neg 
(gnc:split-get-value s))))))))

                               ((split-account-type? s 'expense)
                                (brokeragecoll 'add currency 
(gnc:split-get-value s)))

                               ((split-account-type? s 'income)
                                (dividendcoll 'add currency 
(gnc:split-get-value s)))
                               )
                             )
                           (gnc:transaction-get-splits parent)
                           )
                          )
                      )
                    )
                  (gnc:account-get-split-list current)
                  )

                 ;;                (gnc:debug "totalunits" totalunits)
                 ;;                (gnc:debug "totalunityears" 
totalunityears)

                 (gaincoll 'merge moneyoutcoll #f)
                 (gaincoll 'add (gnc:gnc-monetary-commodity value) 
(gnc:gnc-monetary-amount value))
                 (gaincoll 'merge moneyincoll #f)

                 (gnc:price-list-destroy price-list)

                 (if (or include-empty (not (gnc:numeric-zero-p units)))
                     (begin
                       (total-value    'add (gnc:gnc-monetary-commodity 
value) (gnc:gnc-monetary-amount value))
                       (total-moneyin  'merge moneyincoll #f)
                       (total-moneyout 'merge moneyoutcoll #f)
                       (total-gain     'merge gaincoll #f)

                       ;;(gnc:warn ">name: " (gnc:account-get-name 
current))
                       ;;(if price (gnc:warn ">price: " 
(gnc:numeric-to-double (gnc:gnc-monetary-amount rate))))

                       (cons (list current
                                   ticker-symbol
                                   listing
                                   units
                                   rate
                                   value
                                   (gnc:monetary-neg 
(gnc:sum-collector-commodity moneyincoll currency exchange-fn))
                                   (gnc:sum-collector-commodity 
moneyoutcoll currency exchange-fn)
                                   (gnc:sum-collector-commodity gaincoll 
currency exchange-fn)
                                   (* 100 (/ (gnc:numeric-to-double 
(cadr (gaincoll 'getpair currency #f)))
                                             (gnc:numeric-to-double 
(cadr (moneyincoll 'getpair currency #t)))))
                                   price)
                             (table-add-stock-rows-internal rest)))

                     (table-add-stock-rows-internal rest)))
               ))

         (set! work-to-do (gnc:accounts-count-splits accounts)) ; 
#splits as progress indicator

         (table-add-stock-rows-internal accounts)

         ))

     ;; add one row with stock-computed values to the HTML table
     (define (table-add-stock-row-html table share-print-info odd-row?
                                       current ticker-symbol listing 
units
                                       rate value money-in money-out 
gain return price)
       (let* ((row-style (if odd-row? "normal-row" "alternate-row"))
              (odd-row? (not odd-row?)))
         (gnc:html-table-append-row/markup!
          table
          row-style
          (list (gnc:html-account-anchor current)
                ticker-symbol
                listing
                (gnc:make-html-table-header-cell/markup "number-cell" 
(gnc:amount->string units share-print-info))
                (gnc:make-html-table-header-cell/markup
                 "number-cell"
                 (if price
                     (gnc:html-price-anchor price rate)
                     #f))
                (gnc:make-html-table-header-cell/markup "number-cell" 
value)
                (gnc:make-html-table-header-cell/markup "number-cell" 
money-in)
                (gnc:make-html-table-header-cell/markup "number-cell" 
money-out)
                (gnc:make-html-table-header-cell/markup "number-cell" 
gain)
                (gnc:make-html-table-header-cell/markup "number-cell" 
(sprintf #f "%.3f%%" return))
                ))))

     ;; add all computed values off all stocks selected to the HTML table
     (define (table-add-stock-rows-html table account-totals)
       (let*
           ;; get printing related options first
           ((share-print-info (gnc:share-print-info-places (get-option 
gnc:pagename-general
                                                                       
optname-shares-digits)))
            (c (get-option gnc:pagename-general
                           optname-sort-column))
            (odd-row? #t))

         (for-each
          (lambda (l)
            (apply table-add-stock-row-html table share-print-info 
odd-row? l))
          ;; sort column (c in sort-list compare-less procedure) offsets 
are:
          ;; 0 account
          ;; 1 symbol
          ;; 2 listing
          ;; 3 shares (units)
          ;; 4 price (rate)
          ;; 5 value
          ;; 6 money-in
          ;; 7 money-out
          ;; 8 gain
          ;; 9 total-return
          (sort-list account-totals (cond
                                      ((= c 0)
                                       (lambda (list1 list2)
                                         (if (string<? 
(gnc:account-get-name (list-ref list1 c))
                                                       
(gnc:account-get-name (list-ref list2 c))) #t #f)))
                                      ((and (> c 0) (< c 3))
                                       (lambda (list1 list2)
                                         (if (string<? (list-ref list1 
c) (list-ref list2 c)) #t #f)))
                                      ((= c 3)
                                       (lambda (list1 list2)
                                         (if (< 
(gnc:numeric-to-double(list-ref list1 c))
                                                
(gnc:numeric-to-double(list-ref list2 c))) #t #f)))
                                      ((= c 4)
                                       (lambda (list1 list2)
                                         (if (< (if (list-ref list1 c) 
(gnc:numeric-to-double (gnc:gnc-monetary-amount (list-ref list1 c))) 
0.0)
                                                (if (list-ref list2 c) 
(gnc:numeric-to-double (gnc:gnc-monetary-amount (list-ref list2 c))) 
0.0)) #t #f)))
                                      ((and (> c 4) (< c 9))
                                       (lambda (list1 list2)
                                         (if (< (gnc:numeric-to-double 
(gnc:gnc-monetary-amount (list-ref list1 c)))
                                                (gnc:numeric-to-double 
(gnc:gnc-monetary-amount (list-ref list2 c)))) #t #f)))
                                      ((> c 8)
                                       (lambda (list1 list2)
                                         (if (< (list-ref list1 c) 
(list-ref list2 c)) #t #f)))
                                      )))))

     ;; Tell the user that we're starting.
     (gnc:report-starting reportname)

     ;; The first thing we do is make local variables for all the 
specific
     ;; options in the set of options given to the function. This set 
will
     ;; be generated by the options generator above.
     (let ((to-date       (gnc:date-option-absolute-time
                           (get-option gnc:pagename-general "Date")))
           (accounts      (get-option gnc:pagename-accounts "Accounts"))
           (currency      (get-option gnc:pagename-general "Report 
Currency"))
           (price-source  (get-option gnc:pagename-general
                                      optname-price-source))
           (report-title  (get-option gnc:pagename-general
                                      gnc:optname-reportname))
           (include-empty (get-option gnc:pagename-accounts
                                      optname-zero-shares))
           (include-gains (get-option gnc:pagename-general
                                      optname-include-gains))

           (total-value    (gnc:make-commodity-collector))
           (total-moneyin  (gnc:make-commodity-collector))
           (total-moneyout (gnc:make-commodity-collector))
           (total-gain     (gnc:make-commodity-collector))
           ;; document will be the HTML document that we return.
           (table (gnc:make-html-table))
           (document (gnc:make-html-document)))

       (gnc:html-document-set-title!
        document (string-append
                  report-title
                  (sprintf #f " %s" (gnc:print-date to-date))))

       ;;    (gnc:debug "accounts" accounts)
       (if (not (null? accounts))
           ; at least one account selected
           (let* ((exchange-fn
                   (case price-source
                     ('pricedb-latest
                       (lambda (foreign domestic date)
                         (gnc:exchange-by-pricedb-latest foreign 
domestic)))
                     ('pricedb-nearest gnc:exchange-by-pricedb-nearest)))
                  (pricedb (gnc:book-get-pricedb (gnc:get-current-book)))
                  (price-fn
                   (case price-source
                     ('pricedb-latest
                       (lambda (foreign date)
                         (gnc:pricedb-lookup-latest-any-currency pricedb 
foreign)))
                     ('pricedb-nearest
                       (lambda (foreign date)
                         
(gnc:pricedb-lookup-nearest-in-time-any-currency pricedb foreign 
date))))))

             (gnc:html-table-set-col-headers!
              table
              (list (_ "Account")
                    (_ "Symbol")
                    (_ "Listing")
                    (_ "Shares")
                    (_ "Price")
                    (_ "Value")
                    (_ "Money In")
                    (_ "Money Out")
                    (_ "Gain")
                    (_ "Total Return")))

             (table-add-stock-rows-html table
                                        (table-add-stock-rows accounts 
to-date
                                                              currency 
price-fn exchange-fn
                                                              
include-empty include-gains
                                                              
total-value total-moneyin total-moneyout total-gain))

             (gnc:html-table-append-row/markup!
              table
              "grand-total"
              (list
               (gnc:make-html-table-cell/size
                1 10 (gnc:make-html-text (gnc:html-markup-hr)))))

             (gnc:html-table-append-row/markup!
              table
              "grand-total"
              (list (gnc:make-html-table-cell/markup
                     "total-label-cell" (_ "Total"))
                    ""
                    ""
                    ""
                    ""
                    (gnc:make-html-table-cell/markup
                     "total-number-cell" (gnc:sum-collector-commodity 
total-value currency exchange-fn))
                    (gnc:make-html-table-cell/markup
                     "total-number-cell" (gnc:monetary-neg 
(gnc:sum-collector-commodity total-moneyin currency exchange-fn)))
                    (gnc:make-html-table-cell/markup
                     "total-number-cell" (gnc:sum-collector-commodity 
total-moneyout currency exchange-fn))
                    (gnc:make-html-table-cell/markup
                     "total-number-cell" (gnc:sum-collector-commodity 
total-gain currency exchange-fn))
                    (gnc:make-html-table-cell/markup
                     "total-number-cell" (sprintf #f "%.2f%%" (* 100 (/ 
(gnc:numeric-to-double (cadr (total-gain 'getpair currency #f)))
                                                                        
(gnc:numeric-to-double (cadr (total-moneyin 'getpair currency #t)))))))
                    ))

             ;;          (total-value
             ;;           'format
             ;;           (lambda (currency amount)
             ;;             (gnc:html-table-append-row/markup!
             ;;              table
             ;;              "grand-total"
             ;;              (list (gnc:make-html-table-cell/markup
             ;;                     "total-label-cell" (_ "Total"))
             ;;                    (gnc:make-html-table-cell/size/markup
             ;;                     1 5 "total-number-cell"
             ;;                     (gnc:make-gnc-monetary currency 
amount)))))
             ;;           #f)

             (gnc:html-document-add-object! document table))

           ;if no accounts selected.
           (gnc:html-document-add-object!
            document
            (gnc:html-make-no-account-warning
             report-title (gnc:report-id report-obj))))

       (gnc:report-finished)
       document)))

(gnc:define-report
  'version 1
  'name reportname
  'menu-path (list gnc:menuname-asset-liability)
  'options-generator options-generator
  'renderer advanced-portfolio-sorted-renderer)




More information about the gnucash-user mailing list