|  | // Copyright 2025 The BoringSSL Authors | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | // you may not use this file except in compliance with the License. | 
|  | // You may obtain a copy of the License at | 
|  | // | 
|  | //     https://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | // See the License for the specific language governing permissions and | 
|  | // limitations under the License. | 
|  |  | 
|  | //go:build gcs | 
|  |  | 
|  | package main | 
|  |  | 
|  | import ( | 
|  | "context" | 
|  | "flag" | 
|  | "io" | 
|  | "log" | 
|  | neturl "net/url" | 
|  | "time" | 
|  |  | 
|  | "cloud.google.com/go/storage" | 
|  | "google.golang.org/api/iterator" | 
|  | ) | 
|  |  | 
|  | var uploadGCSDirectory = flag.String("gcs", "", "GCS path to folder where result files to be uploaded are") | 
|  |  | 
|  | // Uploads results directory from GCS. | 
|  | // Similar to uploadResultsDirectory(). | 
|  | func uploadResultsFromGCS(gcsBucket string, config *Config, sessionTokensCacheDir string) { | 
|  | u, err := neturl.Parse(gcsBucket) | 
|  | if err != nil { | 
|  | log.Fatal(err) | 
|  | } | 
|  | bucket := u.Host | 
|  | folder := trimLeadingSlash(addTrailingSlash(u.Path)) | 
|  | sessionID, err := getLastDigitDir(folder) | 
|  | if err != nil { | 
|  | log.Fatal(err) | 
|  | } | 
|  |  | 
|  | ctx := context.Background() | 
|  | client, err := storage.NewClient(ctx) | 
|  | if err != nil { | 
|  | log.Fatalf("Failed to create client: %v", err) | 
|  | } | 
|  | defer client.Close() | 
|  | ctx, cancel := context.WithTimeout(ctx, time.Second*10) | 
|  | defer cancel() | 
|  |  | 
|  | // Access bucket and identify all objects. | 
|  | // Objects include the folder. | 
|  | it := client.Bucket(bucket).Objects(ctx, &storage.Query{ | 
|  | Prefix:    folder, | 
|  | Delimiter: "/", | 
|  | }) | 
|  |  | 
|  | var results []nistUploadResult | 
|  | // Go through each object (noting GCS stores objects, not files) | 
|  | for { | 
|  | attrs, err := it.Next() | 
|  | if err == iterator.Done { | 
|  | break | 
|  | } else if err != nil { | 
|  | log.Fatalf("Unable to read bucket: %s", err) | 
|  | } | 
|  | rc, err := client.Bucket(bucket).Object(attrs.Name).NewReader(ctx) | 
|  | if err != nil { | 
|  | log.Fatalf("unable to open object from bucket %q, object %q: %v", bucket, attrs.Name, err) | 
|  | return | 
|  | } | 
|  | defer rc.Close() | 
|  | content, err := io.ReadAll(rc) | 
|  | if err != nil { | 
|  | log.Fatalf("unable to read contents of object %q: %v", attrs.Name, err) | 
|  | } | 
|  | if len(content) == 0 { | 
|  | log.Printf("object (gs://%s/%s) is a \"folder\" or empty.", bucket, attrs.Name) | 
|  | continue | 
|  | } | 
|  |  | 
|  | results = processResultContent(results, content, sessionID, attrs.Name) | 
|  | } | 
|  |  | 
|  | uploadResults(results, sessionID, config, sessionTokensCacheDir) | 
|  | } | 
|  |  | 
|  | func handleGCSFlag(config *Config, sessionTokensCacheDir string) bool { | 
|  | if len(*uploadGCSDirectory) > 0 { | 
|  | uploadResultsFromGCS(*uploadGCSDirectory, config, sessionTokensCacheDir) | 
|  | return true | 
|  | } | 
|  | return false | 
|  | } |