17678: implement review feedback.
[arvados.git] / lib / costanalyzer / costanalyzer.go
index c86e267695c7b6d05a3ed3b7f34a14cc3489a233..402e0ec81a160e677a1aefb8df82d589f01c3e20 100644 (file)
@@ -5,7 +5,6 @@
 package costanalyzer
 
 import (
 package costanalyzer
 
 import (
-       "bytes"
        "encoding/json"
        "errors"
        "flag"
        "encoding/json"
        "errors"
        "flag"
@@ -16,7 +15,6 @@ import (
        "git.arvados.org/arvados.git/sdk/go/keepclient"
        "io"
        "io/ioutil"
        "git.arvados.org/arvados.git/sdk/go/keepclient"
        "io"
        "io/ioutil"
-       "log"
        "net/http"
        "os"
        "strconv"
        "net/http"
        "os"
        "strconv"
@@ -26,54 +24,17 @@ import (
        "github.com/sirupsen/logrus"
 )
 
        "github.com/sirupsen/logrus"
 )
 
-// Dict is a helper type so we don't have to write out 'map[string]interface{}' every time.
-type Dict map[string]interface{}
-
-// LegacyNodeInfo is a struct for records created by Arvados Node Manager (Arvados <= 1.4.3)
-// Example:
-// {
-//    "total_cpu_cores":2,
-//    "total_scratch_mb":33770,
-//    "cloud_node":
-//      {
-//        "price":0.1,
-//        "size":"m4.large"
-//      },
-//     "total_ram_mb":7986
-// }
-type LegacyNodeInfo struct {
-       CPUCores  int64           `json:"total_cpu_cores"`
-       ScratchMb int64           `json:"total_scratch_mb"`
-       RAMMb     int64           `json:"total_ram_mb"`
-       CloudNode LegacyCloudNode `json:"cloud_node"`
-}
-
-// LegacyCloudNode is a struct for records created by Arvados Node Manager (Arvados <= 1.4.3)
-type LegacyCloudNode struct {
-       Price float64 `json:"price"`
-       Size  string  `json:"size"`
-}
-
-// Node is a struct for records created by Arvados Dispatch Cloud (Arvados >= 2.0.0)
-// Example:
-// {
-//    "Name": "Standard_D1_v2",
-//    "ProviderType": "Standard_D1_v2",
-//    "VCPUs": 1,
-//    "RAM": 3584000000,
-//    "Scratch": 50000000000,
-//    "IncludedScratch": 50000000000,
-//    "AddedScratch": 0,
-//    "Price": 0.057,
-//    "Preemptible": false
-//}
-type Node struct {
-       VCPUs        int64
-       Scratch      int64
-       RAM          int64
-       Price        float64
-       Name         string
+type nodeInfo struct {
+       // Legacy (records created by Arvados Node Manager with Arvados <= 1.4.3)
+       Properties struct {
+               CloudNode struct {
+                       Price float64
+                       Size  string
+               } `json:"cloud_node"`
+       }
+       // Modern
        ProviderType string
        ProviderType string
+       Price        float64
        Preemptible  bool
 }
 
        Preemptible  bool
 }
 
@@ -84,67 +45,90 @@ func (i *arrayFlags) String() string {
 }
 
 func (i *arrayFlags) Set(value string) error {
 }
 
 func (i *arrayFlags) Set(value string) error {
-       *i = append(*i, value)
+       for _, s := range strings.Split(value, ",") {
+               *i = append(*i, s)
+       }
        return nil
 }
 
        return nil
 }
 
-func parseFlags(prog string, args []string, loader *config.Loader, logger *logrus.Logger, stderr io.Writer) (exitCode int, uuids arrayFlags, resultsDir string) {
+func parseFlags(prog string, args []string, loader *config.Loader, logger *logrus.Logger, stderr io.Writer) (exitCode int, uuids arrayFlags, resultsDir string, cache bool, err error) {
        flags := flag.NewFlagSet("", flag.ContinueOnError)
        flags.SetOutput(stderr)
        flags.Usage = func() {
                fmt.Fprintf(flags.Output(), `
 Usage:
        flags := flag.NewFlagSet("", flag.ContinueOnError)
        flags.SetOutput(stderr)
        flags.Usage = func() {
                fmt.Fprintf(flags.Output(), `
 Usage:
-  %s [options ...]
+  %s [options ...] uuid [uuid ...]
 
        This program analyzes the cost of Arvados container requests. For each uuid
        supplied, it creates a CSV report that lists all the containers used to
        fulfill the container request, together with the machine type and cost of
 
        This program analyzes the cost of Arvados container requests. For each uuid
        supplied, it creates a CSV report that lists all the containers used to
        fulfill the container request, together with the machine type and cost of
-       each container.
+       each container. At least one uuid must be specified.
 
        When supplied with the uuid of a container request, it will calculate the
 
        When supplied with the uuid of a container request, it will calculate the
-       cost of that container request and all its children. When suplied with a
-       project uuid or when supplied with multiple container request uuids, it will
-       create a CSV report for each supplied uuid, as well as a CSV file with
-       aggregate cost accounting for all supplied uuids. The aggregate cost report
-       takes container reuse into account: if a container was reused between several
-       container requests, its cost will only be counted once.
-
-       To get the node costs, the progam queries the Arvados API for current cost
-       data for each node type used. This means that the reported cost always
-       reflects the cost data as currently defined in the Arvados API configuration
-       file.
+       cost of that container request and all its children.
+
+       When supplied with the uuid of a collection, it will see if there is a
+       container_request uuid in the properties of the collection, and if so, it
+       will calculate the cost of that container request and all its children.
+
+       When supplied with a project uuid or when supplied with multiple container
+       request or collection uuids, it will create a CSV report for each supplied
+       uuid, as well as a CSV file with aggregate cost accounting for all supplied
+       uuids. The aggregate cost report takes container reuse into account: if a
+       container was reused between several container requests, its cost will only
+       be counted once.
 
        Caveats:
 
        Caveats:
-       - the Arvados API configuration cost data may be out of sync with the cloud
-       provider.
-       - when generating reports for older container requests, the cost data in the
-       Arvados API configuration file may have changed since the container request
-       was fulfilled. This program uses the cost data stored at the time of the
+
+       - This program uses the cost data from config.yml at the time of the
        execution of the container, stored in the 'node.json' file in its log
        execution of the container, stored in the 'node.json' file in its log
-       collection.
+       collection. If the cost data was not correctly configured at the time the
+       container was executed, the output from this program will be incorrect.
+
+       - If a container was run on a preemptible ("spot") instance, the cost data
+       reported by this program may be wildly inaccurate, because it does not have
+       access to the spot pricing in effect for the node then the container ran. The
+       UUID report file that is generated when the '-output' option is specified has
+       a column that indicates the preemptible state of the instance that ran the
+       container.
+
+       - This program does not take into account overhead costs like the time spent
+       starting and stopping compute nodes that run containers, the cost of the
+       permanent cloud nodes that provide the Arvados services, the cost of data
+       stored in Arvados, etc.
+
+       - When provided with a project uuid, subprojects will not be considered.
 
        In order to get the data for the uuids supplied, the ARVADOS_API_HOST and
        ARVADOS_API_TOKEN environment variables must be set.
 
 
        In order to get the data for the uuids supplied, the ARVADOS_API_HOST and
        ARVADOS_API_TOKEN environment variables must be set.
 
+       This program prints the total dollar amount from the aggregate cost
+       accounting across all provided uuids on stdout.
+
+       When the '-output' option is specified, a set of CSV files with cost details
+       will be written to the provided directory.
+
 Options:
 `, prog)
                flags.PrintDefaults()
        }
 Options:
 `, prog)
                flags.PrintDefaults()
        }
-       loglevel := flags.String("log-level", "info", "logging level (debug, info, ...)")
-       resultsDir = *flags.String("output", "results", "output directory for the CSV reports")
-       flags.Var(&uuids, "uuid", "Toplevel project or container request uuid. May be specified more than once.")
-       err := flags.Parse(args)
+       loglevel := flags.String("log-level", "info", "logging `level` (debug, info, ...)")
+       flags.StringVar(&resultsDir, "output", "", "output `directory` for the CSV reports")
+       flags.BoolVar(&cache, "cache", true, "create and use a local disk cache of Arvados objects")
+       err = flags.Parse(args)
        if err == flag.ErrHelp {
        if err == flag.ErrHelp {
+               err = nil
                exitCode = 1
                return
        } else if err != nil {
                exitCode = 2
                return
        }
                exitCode = 1
                return
        } else if err != nil {
                exitCode = 2
                return
        }
+       uuids = flags.Args()
 
        if len(uuids) < 1 {
 
        if len(uuids) < 1 {
-               logger.Errorf("Error: no uuid(s) provided")
                flags.Usage()
                flags.Usage()
+               err = fmt.Errorf("error: no uuid(s) provided")
                exitCode = 2
                return
        }
                exitCode = 2
                return
        }
@@ -155,6 +139,9 @@ Options:
                return
        }
        logger.SetLevel(lvl)
                return
        }
        logger.SetLevel(lvl)
+       if !cache {
+               logger.Debug("Caching disabled\n")
+       }
        return
 }
 
        return
 }
 
@@ -163,217 +150,196 @@ func ensureDirectory(logger *logrus.Logger, dir string) (err error) {
        if os.IsNotExist(err) {
                err = os.MkdirAll(dir, 0700)
                if err != nil {
        if os.IsNotExist(err) {
                err = os.MkdirAll(dir, 0700)
                if err != nil {
-                       logger.Errorf("Error creating directory %s: %s\n", dir, err.Error())
-                       return
+                       return fmt.Errorf("error creating directory %s: %s", dir, err.Error())
                }
        } else {
                if !statData.IsDir() {
                }
        } else {
                if !statData.IsDir() {
-                       logger.Errorf("The path %s is not a directory\n", dir)
-                       return
+                       return fmt.Errorf("the path %s is not a directory", dir)
                }
        }
        return
 }
 
                }
        }
        return
 }
 
-func addContainerLine(logger *logrus.Logger, node interface{}, cr Dict, container Dict) (csv string, cost float64) {
-       csv = cr["uuid"].(string) + ","
-       csv += cr["name"].(string) + ","
-       csv += container["uuid"].(string) + ","
-       csv += container["state"].(string) + ","
-       if container["started_at"] != nil {
-               csv += container["started_at"].(string) + ","
+func addContainerLine(logger *logrus.Logger, node nodeInfo, cr arvados.ContainerRequest, container arvados.Container) (csv string, cost float64) {
+       csv = cr.UUID + ","
+       csv += cr.Name + ","
+       csv += container.UUID + ","
+       csv += string(container.State) + ","
+       if container.StartedAt != nil {
+               csv += container.StartedAt.String() + ","
        } else {
                csv += ","
        }
 
        var delta time.Duration
        } else {
                csv += ","
        }
 
        var delta time.Duration
-       if container["finished_at"] != nil {
-               csv += container["finished_at"].(string) + ","
-               finishedTimestamp, err := time.Parse("2006-01-02T15:04:05.000000000Z", container["finished_at"].(string))
-               if err != nil {
-                       fmt.Println(err)
-               }
-               startedTimestamp, err := time.Parse("2006-01-02T15:04:05.000000000Z", container["started_at"].(string))
-               if err != nil {
-                       fmt.Println(err)
-               }
-               delta = finishedTimestamp.Sub(startedTimestamp)
+       if container.FinishedAt != nil {
+               csv += container.FinishedAt.String() + ","
+               delta = container.FinishedAt.Sub(*container.StartedAt)
                csv += strconv.FormatFloat(delta.Seconds(), 'f', 0, 64) + ","
        } else {
                csv += ",,"
        }
        var price float64
        var size string
                csv += strconv.FormatFloat(delta.Seconds(), 'f', 0, 64) + ","
        } else {
                csv += ",,"
        }
        var price float64
        var size string
-       switch n := node.(type) {
-       case Node:
-               price = n.Price
-               size = n.ProviderType
-       case LegacyNodeInfo:
-               price = n.CloudNode.Price
-               size = n.CloudNode.Size
-       default:
-               logger.Warn("WARNING: unknown node type found!")
+       if node.Properties.CloudNode.Price != 0 {
+               price = node.Properties.CloudNode.Price
+               size = node.Properties.CloudNode.Size
+       } else {
+               price = node.Price
+               size = node.ProviderType
        }
        cost = delta.Seconds() / 3600 * price
        }
        cost = delta.Seconds() / 3600 * price
-       csv += size + "," + strconv.FormatFloat(price, 'f', 8, 64) + "," + strconv.FormatFloat(cost, 'f', 8, 64) + "\n"
+       csv += size + "," + fmt.Sprintf("%+v", node.Preemptible) + "," + strconv.FormatFloat(price, 'f', 8, 64) + "," + strconv.FormatFloat(cost, 'f', 8, 64) + "\n"
        return
 }
 
        return
 }
 
-func loadCachedObject(logger *logrus.Logger, file string, uuid string) (reload bool, object Dict) {
+func loadCachedObject(logger *logrus.Logger, file string, uuid string, object interface{}) (reload bool) {
        reload = true
        reload = true
+       if strings.Contains(uuid, "-j7d0g-") || strings.Contains(uuid, "-4zz18-") {
+               // We do not cache projects or collections, they have no final state
+               return
+       }
        // See if we have a cached copy of this object
        // See if we have a cached copy of this object
-       if _, err := os.Stat(file); err == nil {
-               data, err := ioutil.ReadFile(file)
-               if err != nil {
-                       logger.Errorf("error reading %q: %s", file, err)
-                       return
-               }
-               err = json.Unmarshal(data, &object)
-               if err != nil {
-                       logger.Errorf("failed to unmarshal json: %s: %s", data, err)
-                       return
-               }
+       _, err := os.Stat(file)
+       if err != nil {
+               return
+       }
+       data, err := ioutil.ReadFile(file)
+       if err != nil {
+               logger.Errorf("error reading %q: %s", file, err)
+               return
+       }
+       err = json.Unmarshal(data, &object)
+       if err != nil {
+               logger.Errorf("failed to unmarshal json: %s: %s", data, err)
+               return
+       }
 
 
-               // See if it is in a final state, if that makes sense
-               // Projects (j7d0g) do not have state so they should always be reloaded
-               if !strings.Contains(uuid, "-j7d0g-") {
-                       if object["state"].(string) == "Complete" || object["state"].(string) == "Failed" {
-                               reload = false
-                               logger.Debugf("Loaded object %s from local cache (%s)\n", uuid, file)
-                               return
-                       }
+       // See if it is in a final state, if that makes sense
+       switch v := object.(type) {
+       case *arvados.ContainerRequest:
+               if v.State == arvados.ContainerRequestStateFinal {
+                       reload = false
+                       logger.Debugf("Loaded object %s from local cache (%s)\n", uuid, file)
+               }
+       case *arvados.Container:
+               if v.State == arvados.ContainerStateComplete || v.State == arvados.ContainerStateCancelled {
+                       reload = false
+                       logger.Debugf("Loaded object %s from local cache (%s)\n", uuid, file)
                }
        }
        return
 }
 
 // Load an Arvados object.
                }
        }
        return
 }
 
 // Load an Arvados object.
-func loadObject(logger *logrus.Logger, arv *arvadosclient.ArvadosClient, path string, uuid string) (object Dict, err error) {
-       err = ensureDirectory(logger, path)
-       if err != nil {
-               return
-       }
-
-       file := path + "/" + uuid + ".json"
+func loadObject(logger *logrus.Logger, ac *arvados.Client, path string, uuid string, cache bool, object interface{}) (err error) {
+       file := uuid + ".json"
 
        var reload bool
 
        var reload bool
-       reload, object = loadCachedObject(logger, file, uuid)
+       var cacheDir string
 
 
-       if reload {
-               var err error
-               if strings.Contains(uuid, "-j7d0g-") {
-                       err = arv.Get("groups", uuid, nil, &object)
-               } else if strings.Contains(uuid, "-xvhdp-") {
-                       err = arv.Get("container_requests", uuid, nil, &object)
-               } else if strings.Contains(uuid, "-dz642-") {
-                       err = arv.Get("containers", uuid, nil, &object)
-               } else {
-                       err = arv.Get("jobs", uuid, nil, &object)
-               }
-               if err != nil {
-                       logger.Fatalf("Error loading object with UUID %q:\n  %s\n", uuid, err)
-               }
-               encoded, err := json.MarshalIndent(object, "", " ")
+       if !cache {
+               reload = true
+       } else {
+               homeDir, err := os.UserHomeDir()
                if err != nil {
                if err != nil {
-                       logger.Fatalf("Error marshaling object with UUID %q:\n  %s\n", uuid, err)
+                       reload = true
+                       logger.Info("Unable to determine current user home directory, not using cache")
+               } else {
+                       cacheDir = homeDir + "/.cache/arvados/costanalyzer/"
+                       err = ensureDirectory(logger, cacheDir)
+                       if err != nil {
+                               reload = true
+                               logger.Infof("Unable to create cache directory at %s, not using cache: %s", cacheDir, err.Error())
+                       } else {
+                               reload = loadCachedObject(logger, cacheDir+file, uuid, object)
+                       }
                }
                }
-               err = ioutil.WriteFile(file, encoded, 0644)
+       }
+       if !reload {
+               return
+       }
+
+       if strings.Contains(uuid, "-j7d0g-") {
+               err = ac.RequestAndDecode(&object, "GET", "arvados/v1/groups/"+uuid, nil, nil)
+       } else if strings.Contains(uuid, "-xvhdp-") {
+               err = ac.RequestAndDecode(&object, "GET", "arvados/v1/container_requests/"+uuid, nil, nil)
+       } else if strings.Contains(uuid, "-dz642-") {
+               err = ac.RequestAndDecode(&object, "GET", "arvados/v1/containers/"+uuid, nil, nil)
+       } else if strings.Contains(uuid, "-4zz18-") {
+               err = ac.RequestAndDecode(&object, "GET", "arvados/v1/collections/"+uuid, nil, nil)
+       } else {
+               err = fmt.Errorf("unsupported object type with UUID %q:\n  %s", uuid, err)
+               return
+       }
+       if err != nil {
+               err = fmt.Errorf("error loading object with UUID %q:\n  %s", uuid, err)
+               return
+       }
+       encoded, err := json.MarshalIndent(object, "", " ")
+       if err != nil {
+               err = fmt.Errorf("error marshaling object with UUID %q:\n  %s", uuid, err)
+               return
+       }
+       if cacheDir != "" {
+               err = ioutil.WriteFile(cacheDir+file, encoded, 0644)
                if err != nil {
                if err != nil {
-                       logger.Fatalf("Error writing file %s:\n  %s\n", file, err)
+                       err = fmt.Errorf("error writing file %s:\n  %s", file, err)
+                       return
                }
        }
        return
 }
 
                }
        }
        return
 }
 
-func getNode(arv *arvadosclient.ArvadosClient, arv2 *arvados.Client, kc *keepclient.KeepClient, itemMap Dict) (node interface{}, err error) {
-       if _, ok := itemMap["log_uuid"]; ok {
-               if itemMap["log_uuid"] == nil {
-                       err = errors.New("No log collection")
-                       return
-               }
-
-               var collection arvados.Collection
-               err = arv.Get("collections", itemMap["log_uuid"].(string), nil, &collection)
-               if err != nil {
-                       err = fmt.Errorf("Error getting collection: %s", err)
-                       return
-               }
+func getNode(arv *arvadosclient.ArvadosClient, ac *arvados.Client, kc *keepclient.KeepClient, cr arvados.ContainerRequest) (node nodeInfo, err error) {
+       if cr.LogUUID == "" {
+               err = errors.New("no log collection")
+               return
+       }
 
 
-               var fs arvados.CollectionFileSystem
-               fs, err = collection.FileSystem(arv2, kc)
-               if err != nil {
-                       err = fmt.Errorf("Error opening collection as filesystem: %s", err)
-                       return
-               }
-               var f http.File
-               f, err = fs.Open("node.json")
-               if err != nil {
-                       err = fmt.Errorf("Error opening file 'node.json' in collection %s: %s", itemMap["log_uuid"].(string), err)
-                       return
-               }
+       var collection arvados.Collection
+       err = ac.RequestAndDecode(&collection, "GET", "arvados/v1/collections/"+cr.LogUUID, nil, nil)
+       if err != nil {
+               err = fmt.Errorf("error getting collection: %s", err)
+               return
+       }
 
 
-               var nodeDict Dict
-               buf := new(bytes.Buffer)
-               _, err = buf.ReadFrom(f)
-               if err != nil {
-                       err = fmt.Errorf("Error reading file 'node.json' in collection %s: %s", itemMap["log_uuid"].(string), err)
-                       return
-               }
-               contents := buf.String()
-               f.Close()
+       var fs arvados.CollectionFileSystem
+       fs, err = collection.FileSystem(ac, kc)
+       if err != nil {
+               err = fmt.Errorf("error opening collection as filesystem: %s", err)
+               return
+       }
+       var f http.File
+       f, err = fs.Open("node.json")
+       if err != nil {
+               err = fmt.Errorf("error opening file 'node.json' in collection %s: %s", cr.LogUUID, err)
+               return
+       }
 
 
-               err = json.Unmarshal([]byte(contents), &nodeDict)
-               if err != nil {
-                       err = fmt.Errorf("Error unmarshalling: %s", err)
-                       return
-               }
-               if val, ok := nodeDict["properties"]; ok {
-                       var encoded []byte
-                       encoded, err = json.MarshalIndent(val, "", " ")
-                       if err != nil {
-                               err = fmt.Errorf("Error marshalling: %s", err)
-                               return
-                       }
-                       // node is type LegacyNodeInfo
-                       var newNode LegacyNodeInfo
-                       err = json.Unmarshal(encoded, &newNode)
-                       if err != nil {
-                               err = fmt.Errorf("Error unmarshalling: %s", err)
-                               return
-                       }
-                       node = newNode
-               } else {
-                       // node is type Node
-                       var newNode Node
-                       err = json.Unmarshal([]byte(contents), &newNode)
-                       if err != nil {
-                               err = fmt.Errorf("Error unmarshalling: %s", err)
-                               return
-                       }
-                       node = newNode
-               }
+       err = json.NewDecoder(f).Decode(&node)
+       if err != nil {
+               err = fmt.Errorf("error reading file 'node.json' in collection %s: %s", cr.LogUUID, err)
+               return
        }
        return
 }
 
        }
        return
 }
 
-func handleProject(logger *logrus.Logger, uuid string, arv *arvadosclient.ArvadosClient, arv2 *arvados.Client, kc *keepclient.KeepClient, resultsDir string) (cost map[string]float64) {
-
+func handleProject(logger *logrus.Logger, uuid string, arv *arvadosclient.ArvadosClient, ac *arvados.Client, kc *keepclient.KeepClient, resultsDir string, cache bool) (cost map[string]float64, err error) {
        cost = make(map[string]float64)
 
        cost = make(map[string]float64)
 
-       project, err := loadObject(logger, arv, resultsDir+"/"+uuid, uuid)
+       var project arvados.Group
+       err = loadObject(logger, ac, uuid, uuid, cache, &project)
        if err != nil {
        if err != nil {
-               logger.Fatalf("Error loading object %s: %s\n", uuid, err.Error())
+               return nil, fmt.Errorf("error loading object %s: %s", uuid, err.Error())
        }
 
        }
 
-       // arv -f uuid container_request list --filters '[["owner_uuid","=","<someuuid>"],["requesting_container_uuid","=",null]]'
-
-       // Now find all container requests that have the container we found above as requesting_container_uuid
        var childCrs map[string]interface{}
        filterset := []arvados.Filter{
                {
                        Attr:     "owner_uuid",
                        Operator: "=",
        var childCrs map[string]interface{}
        filterset := []arvados.Filter{
                {
                        Attr:     "owner_uuid",
                        Operator: "=",
-                       Operand:  project["uuid"].(string),
+                       Operand:  project.UUID,
                },
                {
                        Attr:     "requesting_container_uuid",
                },
                {
                        Attr:     "requesting_container_uuid",
@@ -381,16 +347,23 @@ func handleProject(logger *logrus.Logger, uuid string, arv *arvadosclient.Arvado
                        Operand:  nil,
                },
        }
                        Operand:  nil,
                },
        }
-       err = arv.List("container_requests", arvadosclient.Dict{"filters": filterset, "limit": 10000}, &childCrs)
+       err = ac.RequestAndDecode(&childCrs, "GET", "arvados/v1/container_requests", nil, map[string]interface{}{
+               "filters": filterset,
+               "limit":   10000,
+       })
        if err != nil {
        if err != nil {
-               logger.Fatalf("Error querying container_requests: %s\n", err.Error())
+               return nil, fmt.Errorf("error querying container_requests: %s", err.Error())
        }
        if value, ok := childCrs["items"]; ok {
                logger.Infof("Collecting top level container requests in project %s\n", uuid)
                items := value.([]interface{})
                for _, item := range items {
                        itemMap := item.(map[string]interface{})
        }
        if value, ok := childCrs["items"]; ok {
                logger.Infof("Collecting top level container requests in project %s\n", uuid)
                items := value.([]interface{})
                for _, item := range items {
                        itemMap := item.(map[string]interface{})
-                       for k, v := range generateCrCsv(logger, itemMap["uuid"].(string), arv, arv2, kc, resultsDir) {
+                       crCsv, err := generateCrCsv(logger, itemMap["uuid"].(string), arv, ac, kc, resultsDir, cache)
+                       if err != nil {
+                               return nil, fmt.Errorf("error generating container_request CSV: %s", err.Error())
+                       }
+                       for k, v := range crCsv {
                                cost[k] = v
                        }
                }
                                cost[k] = v
                        }
                }
@@ -400,133 +373,170 @@ func handleProject(logger *logrus.Logger, uuid string, arv *arvadosclient.Arvado
        return
 }
 
        return
 }
 
-func generateCrCsv(logger *logrus.Logger, uuid string, arv *arvadosclient.ArvadosClient, arv2 *arvados.Client, kc *keepclient.KeepClient, resultsDir string) (cost map[string]float64) {
+func generateCrCsv(logger *logrus.Logger, uuid string, arv *arvadosclient.ArvadosClient, ac *arvados.Client, kc *keepclient.KeepClient, resultsDir string, cache bool) (cost map[string]float64, err error) {
 
        cost = make(map[string]float64)
 
 
        cost = make(map[string]float64)
 
-       csv := "CR UUID,CR name,Container UUID,State,Started At,Finished At,Duration in seconds,Compute node type,Hourly node cost,Total cost\n"
+       csv := "CR UUID,CR name,Container UUID,State,Started At,Finished At,Duration in seconds,Compute node type,Preemptible,Hourly node cost,Total cost\n"
        var tmpCsv string
        var tmpTotalCost float64
        var totalCost float64
 
        var tmpCsv string
        var tmpTotalCost float64
        var totalCost float64
 
+       var crUUID = uuid
+       if strings.Contains(uuid, "-4zz18-") {
+               // This is a collection, find the associated container request (if any)
+               var c arvados.Collection
+               err = loadObject(logger, ac, uuid, uuid, cache, &c)
+               if err != nil {
+                       return nil, fmt.Errorf("error loading collection object %s: %s", uuid, err)
+               }
+               value, ok := c.Properties["container_request"]
+               if !ok {
+                       return nil, fmt.Errorf("error: collection %s does not have a 'container_request' property", uuid)
+               }
+               crUUID, ok = value.(string)
+               if !ok {
+                       return nil, fmt.Errorf("error: collection %s does not have a 'container_request' property of the string type", uuid)
+               }
+       }
+
        // This is a container request, find the container
        // This is a container request, find the container
-       cr, err := loadObject(logger, arv, resultsDir+"/"+uuid, uuid)
+       var cr arvados.ContainerRequest
+       err = loadObject(logger, ac, crUUID, crUUID, cache, &cr)
        if err != nil {
        if err != nil {
-               log.Fatalf("Error loading object %s: %s", uuid, err)
+               return nil, fmt.Errorf("error loading cr object %s: %s", uuid, err)
        }
        }
-       container, err := loadObject(logger, arv, resultsDir+"/"+uuid, cr["container_uuid"].(string))
+       var container arvados.Container
+       err = loadObject(logger, ac, crUUID, cr.ContainerUUID, cache, &container)
        if err != nil {
        if err != nil {
-               log.Fatalf("Error loading object %s: %s", cr["container_uuid"].(string), err)
+               return nil, fmt.Errorf("error loading container object %s: %s", cr.ContainerUUID, err)
        }
 
        }
 
-       topNode, err := getNode(arv, arv2, kc, cr)
+       topNode, err := getNode(arv, ac, kc, cr)
        if err != nil {
        if err != nil {
-               log.Fatalf("Error getting node %s: %s\n", cr["uuid"], err)
+               return nil, fmt.Errorf("error getting node %s: %s", cr.UUID, err)
        }
        tmpCsv, totalCost = addContainerLine(logger, topNode, cr, container)
        csv += tmpCsv
        totalCost += tmpTotalCost
        }
        tmpCsv, totalCost = addContainerLine(logger, topNode, cr, container)
        csv += tmpCsv
        totalCost += tmpTotalCost
+       cost[container.UUID] = totalCost
 
 
-       cost[container["uuid"].(string)] = totalCost
-
-       // Now find all container requests that have the container we found above as requesting_container_uuid
-       var childCrs map[string]interface{}
+       // Find all container requests that have the container we found above as requesting_container_uuid
+       var childCrs arvados.ContainerRequestList
        filterset := []arvados.Filter{
                {
                        Attr:     "requesting_container_uuid",
                        Operator: "=",
        filterset := []arvados.Filter{
                {
                        Attr:     "requesting_container_uuid",
                        Operator: "=",
-                       Operand:  container["uuid"].(string),
+                       Operand:  container.UUID,
                }}
                }}
-       err = arv.List("container_requests", arvadosclient.Dict{"filters": filterset, "limit": 10000}, &childCrs)
+       err = ac.RequestAndDecode(&childCrs, "GET", "arvados/v1/container_requests", nil, map[string]interface{}{
+               "filters": filterset,
+               "limit":   10000,
+       })
        if err != nil {
        if err != nil {
-               log.Fatal("error querying container_requests", err.Error())
+               return nil, fmt.Errorf("error querying container_requests: %s", err.Error())
        }
        }
-       if value, ok := childCrs["items"]; ok {
-               logger.Infof("Collecting child containers for container request %s", uuid)
-               items := value.([]interface{})
-               for _, item := range items {
-                       logger.Info(".")
-                       itemMap := item.(map[string]interface{})
-                       node, err := getNode(arv, arv2, kc, itemMap)
-                       if err != nil {
-                               log.Fatalf("Error getting node %s: %s\n", itemMap["uuid"], err)
-                       }
-                       logger.Debug("\nChild container: " + itemMap["container_uuid"].(string) + "\n")
-                       c2, err := loadObject(logger, arv, resultsDir+"/"+uuid, itemMap["container_uuid"].(string))
-                       if err != nil {
-                               log.Fatalf("Error loading object %s: %s", cr["container_uuid"].(string), err)
-                       }
-                       tmpCsv, tmpTotalCost = addContainerLine(logger, node, itemMap, c2)
-                       cost[itemMap["container_uuid"].(string)] = tmpTotalCost
-                       csv += tmpCsv
-                       totalCost += tmpTotalCost
+       logger.Infof("Collecting child containers for container request %s", crUUID)
+       for _, cr2 := range childCrs.Items {
+               logger.Info(".")
+               node, err := getNode(arv, ac, kc, cr2)
+               if err != nil {
+                       return nil, fmt.Errorf("error getting node %s: %s", cr2.UUID, err)
+               }
+               logger.Debug("\nChild container: " + cr2.ContainerUUID + "\n")
+               var c2 arvados.Container
+               err = loadObject(logger, ac, cr.UUID, cr2.ContainerUUID, cache, &c2)
+               if err != nil {
+                       return nil, fmt.Errorf("error loading object %s: %s", cr2.ContainerUUID, err)
                }
                }
+               tmpCsv, tmpTotalCost = addContainerLine(logger, node, cr2, c2)
+               cost[cr2.ContainerUUID] = tmpTotalCost
+               csv += tmpCsv
+               totalCost += tmpTotalCost
        }
        logger.Info(" done\n")
 
        csv += "TOTAL,,,,,,,,," + strconv.FormatFloat(totalCost, 'f', 8, 64) + "\n"
 
        }
        logger.Info(" done\n")
 
        csv += "TOTAL,,,,,,,,," + strconv.FormatFloat(totalCost, 'f', 8, 64) + "\n"
 
-       // Write the resulting CSV file
-       fName := resultsDir + "/" + uuid + ".csv"
-       err = ioutil.WriteFile(fName, []byte(csv), 0644)
-       if err != nil {
-               logger.Errorf("Error writing file with path %s: %s\n", fName, err.Error())
-               os.Exit(1)
+       if resultsDir != "" {
+               // Write the resulting CSV file
+               fName := resultsDir + "/" + crUUID + ".csv"
+               err = ioutil.WriteFile(fName, []byte(csv), 0644)
+               if err != nil {
+                       return nil, fmt.Errorf("error writing file with path %s: %s", fName, err.Error())
+               }
+               logger.Infof("\nUUID report in %s\n\n", fName)
        }
 
        return
 }
 
        }
 
        return
 }
 
-func costanalyzer(prog string, args []string, loader *config.Loader, logger *logrus.Logger, stdout, stderr io.Writer) (exitcode int) {
-       exitcode, uuids, resultsDir := parseFlags(prog, args, loader, logger, stderr)
+func costanalyzer(prog string, args []string, loader *config.Loader, logger *logrus.Logger, stdout, stderr io.Writer) (exitcode int, err error) {
+       exitcode, uuids, resultsDir, cache, err := parseFlags(prog, args, loader, logger, stderr)
        if exitcode != 0 {
                return
        }
        if exitcode != 0 {
                return
        }
-       err := ensureDirectory(logger, resultsDir)
-       if err != nil {
-               exitcode = 3
-               return
+       if resultsDir != "" {
+               err = ensureDirectory(logger, resultsDir)
+               if err != nil {
+                       exitcode = 3
+                       return
+               }
        }
 
        // Arvados Client setup
        arv, err := arvadosclient.MakeArvadosClient()
        if err != nil {
        }
 
        // Arvados Client setup
        arv, err := arvadosclient.MakeArvadosClient()
        if err != nil {
-               logger.Errorf("error creating Arvados object: %s", err)
-               os.Exit(1)
+               err = fmt.Errorf("error creating Arvados object: %s", err)
+               exitcode = 1
+               return
        }
        kc, err := keepclient.MakeKeepClient(arv)
        if err != nil {
        }
        kc, err := keepclient.MakeKeepClient(arv)
        if err != nil {
-               logger.Errorf("error creating Keep object: %s", err)
-               os.Exit(1)
+               err = fmt.Errorf("error creating Keep object: %s", err)
+               exitcode = 1
+               return
        }
 
        }
 
-       arv2 := arvados.NewClientFromEnv()
+       ac := arvados.NewClientFromEnv()
 
        cost := make(map[string]float64)
        for _, uuid := range uuids {
                if strings.Contains(uuid, "-j7d0g-") {
                        // This is a project (group)
 
        cost := make(map[string]float64)
        for _, uuid := range uuids {
                if strings.Contains(uuid, "-j7d0g-") {
                        // This is a project (group)
-                       for k, v := range handleProject(logger, uuid, arv, arv2, kc, resultsDir) {
+                       cost, err = handleProject(logger, uuid, arv, ac, kc, resultsDir, cache)
+                       if err != nil {
+                               exitcode = 1
+                               return
+                       }
+                       for k, v := range cost {
                                cost[k] = v
                        }
                                cost[k] = v
                        }
-               } else if strings.Contains(uuid, "-xvhdp-") {
+               } else if strings.Contains(uuid, "-xvhdp-") || strings.Contains(uuid, "-4zz18-") {
                        // This is a container request
                        // This is a container request
-                       for k, v := range generateCrCsv(logger, uuid, arv, arv2, kc, resultsDir) {
+                       var crCsv map[string]float64
+                       crCsv, err = generateCrCsv(logger, uuid, arv, ac, kc, resultsDir, cache)
+                       if err != nil {
+                               err = fmt.Errorf("error generating CSV for uuid %s: %s", uuid, err.Error())
+                               exitcode = 2
+                               return
+                       }
+                       for k, v := range crCsv {
                                cost[k] = v
                        }
                } else if strings.Contains(uuid, "-tpzed-") {
                        // This is a user. The "Home" project for a user is not a real project.
                        // It is identified by the user uuid. As such, cost analysis for the
                                cost[k] = v
                        }
                } else if strings.Contains(uuid, "-tpzed-") {
                        // This is a user. The "Home" project for a user is not a real project.
                        // It is identified by the user uuid. As such, cost analysis for the
-                       // "Home" project is not supported by this program.
-                       logger.Errorf("Cost analysis is not supported for the 'Home' project: %s", uuid)
+                       // "Home" project is not supported by this program. Skip this uuid, but
+                       // keep going.
+                       logger.Errorf("cost analysis is not supported for the 'Home' project: %s", uuid)
+               } else {
+                       logger.Errorf("this argument does not look like a uuid: %s\n", uuid)
+                       exitcode = 3
+                       return
                }
        }
 
                }
        }
 
-       logger.Info("\n")
-       for k := range cost {
-               logger.Infof("Uuid report in %s/%s.csv\n", resultsDir, k)
-       }
-
        if len(cost) == 0 {
                logger.Info("Nothing to do!\n")
                return
        if len(cost) == 0 {
                logger.Info("Nothing to do!\n")
                return
@@ -547,14 +557,20 @@ func costanalyzer(prog string, args []string, loader *config.Loader, logger *log
 
        csv += "TOTAL," + strconv.FormatFloat(total, 'f', 8, 64) + "\n"
 
 
        csv += "TOTAL," + strconv.FormatFloat(total, 'f', 8, 64) + "\n"
 
-       // Write the resulting CSV file
-       aFile := resultsDir + "/" + time.Now().Format("2006-01-02-15-04-05") + "-aggregate-costaccounting.csv"
-       err = ioutil.WriteFile(aFile, []byte(csv), 0644)
-       if err != nil {
-               logger.Errorf("Error writing file with path %s: %s\n", aFile, err.Error())
-               os.Exit(1)
-       } else {
-               logger.Infof("\nAggregate cost accounting for all supplied uuids in %s\n", aFile)
+       if resultsDir != "" {
+               // Write the resulting CSV file
+               aFile := resultsDir + "/" + time.Now().Format("2006-01-02-15-04-05") + "-aggregate-costaccounting.csv"
+               err = ioutil.WriteFile(aFile, []byte(csv), 0644)
+               if err != nil {
+                       err = fmt.Errorf("error writing file with path %s: %s", aFile, err.Error())
+                       exitcode = 1
+                       return
+               }
+               logger.Infof("Aggregate cost accounting for all supplied uuids in %s\n", aFile)
        }
        }
+
+       // Output the total dollar amount on stdout
+       fmt.Fprintf(stdout, "%s\n", strconv.FormatFloat(total, 'f', 8, 64))
+
        return
 }
        return
 }