UNPKG

@atomist/sdm

Version:

Atomist Software Delivery Machine SDK

113 lines (100 loc) 4.96 kB
/* * Copyright © 2020 Atomist, Inc. * * 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 * * http://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. */ import { doWithRetry } from "@atomist/automation-client/lib/util/retry"; import { Storage } from "@google-cloud/storage"; import { GoalInvocation } from "../../api/goal/GoalInvocation"; import { CacheConfiguration } from "../../api/machine/SoftwareDeliveryMachineOptions"; import { GoalCacheArchiveStore } from "../../core/goal/cache/CompressingGoalCache"; export interface GoogleCloudStorageCacheConfiguration extends CacheConfiguration { cache?: { /** * Google Cloud Storage bucket to perist cache entries to. If * not provided, it defaults to * "sdm-WORKSPACE_ID-SDM_NAME-goal-cache", with "WORKSPACE_ID" * replaced with your Atomist workspace ID and "SDM_NAME" * replaced with the name of the running SDM, converting * letters to lower case and removing all characters that are * not letters, numbers, and dashes (-). It makes no attempt * to create this bucket, so make sure it exists before trying * to use it. */ bucket?: string; /** Set to true to enable goal input/output caching */ enabled?: boolean; /** Path prefix, defaults to "goal-cache". */ path?: string; }; } type GcsOp = (s: Storage, b: string, p: string) => Promise<any>; export type CacheConfig = Required<Required<GoogleCloudStorageCacheConfiguration>["cache"]>; /** * Goal archive store that stores the compressed archives in a Google * Cloud Storage bucket. All failures are caught and logged. If * retrieval fails, the error is rethrown so the cache-miss listeners * will be invoked. */ export class GoogleCloudStorageGoalCacheArchiveStore implements GoalCacheArchiveStore { public async store(gi: GoalInvocation, classifier: string, archivePath: string): Promise<string> { return this.gcs(gi, classifier, async (storage, bucket, cachePath) => storage.bucket(bucket).upload(archivePath, { destination: cachePath, resumable: false, // avoid https://github.com/googleapis/nodejs-storage/issues/909 }), "store"); } public async delete(gi: GoalInvocation, classifier: string): Promise<void> { await this.gcs(gi, classifier, async (storage, bucket, cachePath) => storage.bucket(bucket).file(cachePath).delete(), "delete"); } public async retrieve(gi: GoalInvocation, classifier: string, targetArchivePath: string): Promise<void> { await this.gcs(gi, classifier, async (storage, bucket, cachePath) => storage.bucket(bucket).file(cachePath).download({ destination: targetArchivePath, }), "retrieve"); } private async gcs(gi: GoalInvocation, classifier: string, op: GcsOp, verb: string): Promise<string> { const cacheConfig = getCacheConfig(gi); const cachePath = getCachePath(cacheConfig, classifier); const storage = new Storage(); const objectUri = `gs://${cacheConfig.bucket}/${cachePath}`; const gerund = verb.replace(/e$/, "ing"); try { gi.progressLog.write(`${gerund} cache archive ${objectUri}`); await doWithRetry(() => op(storage, cacheConfig.bucket, cachePath), `${verb} cache archive`); gi.progressLog.write(`${verb}d cache archive ${objectUri}`); return objectUri; } catch (e) { e.message = `Failed to ${verb} cache archive ${objectUri}: ${e.message}`; gi.progressLog.write(e.message); if (verb === "retrieve") { throw e; } } return undefined; } } /** Construct object path for cache configuration and classifier. */ export function getCachePath(cacheConfig: CacheConfig, classifier: string = "default"): string { return [cacheConfig.path, classifier, "cache.tar.gz"].join("/"); } /** * Retrieve cache configuration and populate with default values. */ export function getCacheConfig(gi: GoalInvocation): CacheConfig { const cacheConfig = gi.configuration.sdm.cache || {}; cacheConfig.enabled = cacheConfig.enabled || false; cacheConfig.bucket = cacheConfig.bucket || `sdm-${gi.context.workspaceId}-${gi.configuration.name}-goal-cache`.toLowerCase().replace(/[^-a-z0-9]*/g, "") .replace(/--+/g, "-"); cacheConfig.path = cacheConfig.path || "goal-cache"; return cacheConfig; }