UNPKG

21.7 kBJavaScriptView Raw
1import * as fs from 'fs-extra';
2import * as path from 'path';
3import { extract } from './extract';
4import { Provider } from './Provider';
5import { upload } from './upload';
6const METADATA_NAME = 'metadata';
7const MAX_SIZE = 1000000000;
8const KEEP_BACKUP_COUNT = 10;
9const extractTime = (prefix, file) => parseInt(file.name.slice(prefix.length).split('/')[1], 10);
10export class GCloudProvider extends Provider {
11 constructor({ environment, options }) {
12 super();
13 this.environment = environment;
14 this.options = options;
15 }
16 async canRestore() {
17 const { time } = await this.getLatestTime();
18 return time !== undefined;
19 }
20 async restore(monitorIn) {
21 const monitor = monitorIn.at('gcloud_provider');
22 const { prefix } = this.options;
23 const { dataPath, tmpPath } = this.environment;
24 const { time, files } = await this.getLatestTime();
25 if (time === undefined) {
26 throw new Error('Cannot restore');
27 }
28 const filePrefix = [prefix, time].join('/');
29 const fileAndPaths = files
30 .filter((file) => file.name.startsWith(filePrefix) && path.basename(file.name) !== METADATA_NAME)
31 .map((file) => ({
32 file,
33 filePath: path.resolve(tmpPath, path.basename(file.name)),
34 }));
35 // tslint:disable-next-line no-loop-statement
36 for (const { file, filePath } of fileAndPaths) {
37 await monitor
38 .withData({ filePath })
39 .captureSpanLog(async () => file.download({ destination: filePath, validation: true }), {
40 name: 'neo_restore_download',
41 });
42 }
43 await Promise.all(fileAndPaths.map(async ({ filePath }) => monitor.withData({ filePath }).captureSpanLog(async () => extract({
44 downloadPath: filePath,
45 dataPath,
46 }), { name: 'neo_restore_extract' })));
47 }
48 async backup(monitorIn) {
49 const monitor = monitorIn.at('gcloud_provider');
50 const { bucket, prefix, keepBackupCount = KEEP_BACKUP_COUNT, maxSizeBytes = MAX_SIZE } = this.options;
51 const { dataPath } = this.environment;
52 const files = await fs.readdir(dataPath);
53 const fileAndStats = await Promise.all(files.map(async (file) => {
54 const stat = await fs.stat(path.resolve(dataPath, file));
55 return { file, stat };
56 }));
57 const mutableFileLists = [];
58 let mutableCurrentFileList = [];
59 let currentSize = 0;
60 // tslint:disable-next-line no-loop-statement
61 for (const { file, stat } of fileAndStats) {
62 if (currentSize > maxSizeBytes) {
63 mutableFileLists.push(mutableCurrentFileList);
64 mutableCurrentFileList = [];
65 currentSize = 0;
66 }
67 mutableCurrentFileList.push(file);
68 currentSize += stat.size;
69 }
70 if (mutableCurrentFileList.length > 0) {
71 mutableFileLists.push(mutableCurrentFileList);
72 }
73 const storage = await this.getStorage();
74 const time = Math.round(Date.now() / 1000);
75 // tslint:disable-next-line no-loop-statement
76 for (const [idx, fileList] of mutableFileLists.entries()) {
77 await monitor.withData({ part: idx }).captureSpanLog(async () => upload({
78 dataPath,
79 write: storage
80 .bucket(bucket)
81 .file([prefix, `${time}`, `storage_part_${idx}.db.tar.gz`].join('/'))
82 .createWriteStream({ validation: true }),
83 fileList,
84 }), { name: 'neo_backup_push' });
85 }
86 await monitor.captureSpanLog(async () => storage
87 .bucket(bucket)
88 .file([prefix, `${time}`, METADATA_NAME].join('/'))
89 .save('', undefined), { name: 'neo_backup_push' });
90 const [fileNames] = await monitor.captureSpanLog(
91 // tslint:disable-next-line no-any no-void-expression no-use-of-empty-return-value
92 async () => storage.bucket(bucket).getFiles({ prefix }), {
93 name: 'neo_backup_list_files',
94 });
95 const times = [...new Set(fileNames.map((file) => extractTime(prefix, file)))];
96 // tslint:disable-next-line no-array-mutation
97 times.sort();
98 const deleteTimes = times.slice(0, -keepBackupCount);
99 await monitor.captureSpanLog(async () => Promise.all(deleteTimes.map(async (deleteTime) => storage.bucket(bucket).deleteFiles({ prefix: [prefix, `${deleteTime}`].join('/') }))), { name: 'neo_backup_delete_old' });
100 }
101 async getLatestTime() {
102 const { bucket, prefix } = this.options;
103 const storage = await this.getStorage();
104 // tslint:disable-next-line no-any no-void-expression no-use-of-empty-return-value
105 const [files] = (await storage.bucket(bucket).getFiles({ prefix }));
106 const metadataTimes = files
107 .filter((file) => path.basename(file.name) === METADATA_NAME)
108 .map((file) => extractTime(prefix, file));
109 // tslint:disable-next-line no-array-mutation
110 metadataTimes.sort();
111 const time = metadataTimes[metadataTimes.length - 1];
112 return { time, files };
113 }
114 async getStorage() {
115 const storage = await import('@google-cloud/storage');
116 // tslint:disable-next-line no-any
117 return new storage.Storage({ projectId: this.options.projectID });
118 }
119}
120
121//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["GCloudProvider.ts"],"names":[],"mappings":"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","file":"neo-one-node-data-backup/src/provider/GCloudProvider.js","sourcesContent":["// tslint:disable-next-line:no-submodule-imports\nimport { File } from '@google-cloud/storage/build/src/file';\nimport { Monitor } from '@neo-one/monitor';\nimport * as fs from 'fs-extra';\nimport * as path from 'path';\nimport { Environment } from '../types';\nimport { extract } from './extract';\nimport { Provider } from './Provider';\nimport { upload } from './upload';\n\nexport interface Options {\n  readonly projectID: string;\n  readonly bucket: string;\n  readonly prefix: string;\n  readonly keepBackupCount?: number;\n  readonly maxSizeBytes?: number;\n}\n\nconst METADATA_NAME = 'metadata';\nconst MAX_SIZE = 1_000_000_000;\nconst KEEP_BACKUP_COUNT = 10;\n\nconst extractTime = (prefix: string, file: File) => parseInt(file.name.slice(prefix.length).split('/')[1], 10);\n\nexport class GCloudProvider extends Provider {\n  private readonly environment: Environment;\n  private readonly options: Options;\n\n  public constructor({ environment, options }: { readonly environment: Environment; readonly options: Options }) {\n    super();\n    this.environment = environment;\n    this.options = options;\n  }\n\n  public async canRestore(): Promise<boolean> {\n    const { time } = await this.getLatestTime();\n\n    return time !== undefined;\n  }\n\n  public async restore(monitorIn: Monitor): Promise<void> {\n    const monitor = monitorIn.at('gcloud_provider');\n    const { prefix } = this.options;\n    const { dataPath, tmpPath } = this.environment;\n\n    const { time, files } = await this.getLatestTime();\n    if (time === undefined) {\n      throw new Error('Cannot restore');\n    }\n\n    const filePrefix = [prefix, time].join('/');\n    const fileAndPaths = files\n      .filter((file) => file.name.startsWith(filePrefix) && path.basename(file.name) !== METADATA_NAME)\n      .map((file) => ({\n        file,\n        filePath: path.resolve(tmpPath, path.basename(file.name)),\n      }));\n\n    // tslint:disable-next-line no-loop-statement\n    for (const { file, filePath } of fileAndPaths) {\n      await monitor\n        .withData({ filePath })\n        .captureSpanLog(async () => file.download({ destination: filePath, validation: true }), {\n          name: 'neo_restore_download',\n        });\n    }\n    await Promise.all(\n      fileAndPaths.map(async ({ filePath }) =>\n        monitor.withData({ filePath }).captureSpanLog(\n          async () =>\n            extract({\n              downloadPath: filePath,\n              dataPath,\n            }),\n          { name: 'neo_restore_extract' },\n        ),\n      ),\n    );\n  }\n\n  public async backup(monitorIn: Monitor): Promise<void> {\n    const monitor = monitorIn.at('gcloud_provider');\n    const { bucket, prefix, keepBackupCount = KEEP_BACKUP_COUNT, maxSizeBytes = MAX_SIZE } = this.options;\n    const { dataPath } = this.environment;\n\n    const files = await fs.readdir(dataPath);\n    const fileAndStats = await Promise.all(\n      files.map(async (file) => {\n        const stat = await fs.stat(path.resolve(dataPath, file));\n\n        return { file, stat };\n      }),\n    );\n\n    const mutableFileLists = [];\n    let mutableCurrentFileList = [];\n    let currentSize = 0;\n    // tslint:disable-next-line no-loop-statement\n    for (const { file, stat } of fileAndStats) {\n      if (currentSize > maxSizeBytes) {\n        mutableFileLists.push(mutableCurrentFileList);\n        mutableCurrentFileList = [];\n        currentSize = 0;\n      }\n\n      mutableCurrentFileList.push(file);\n      currentSize += stat.size;\n    }\n\n    if (mutableCurrentFileList.length > 0) {\n      mutableFileLists.push(mutableCurrentFileList);\n    }\n\n    const storage = await this.getStorage();\n    const time = Math.round(Date.now() / 1000);\n    // tslint:disable-next-line no-loop-statement\n    for (const [idx, fileList] of mutableFileLists.entries()) {\n      await monitor.withData({ part: idx }).captureSpanLog(\n        async () =>\n          upload({\n            dataPath,\n            write: storage\n              .bucket(bucket)\n              .file([prefix, `${time}`, `storage_part_${idx}.db.tar.gz`].join('/'))\n              .createWriteStream({ validation: true }),\n            fileList,\n          }),\n        { name: 'neo_backup_push' },\n      );\n    }\n\n    await monitor.captureSpanLog<Promise<void>>(\n      async () =>\n        storage\n          .bucket(bucket)\n          .file([prefix, `${time}`, METADATA_NAME].join('/'))\n          .save('', undefined),\n      { name: 'neo_backup_push' },\n    );\n\n    const [fileNames] = await monitor.captureSpanLog(\n      // tslint:disable-next-line no-any no-void-expression no-use-of-empty-return-value\n      async () => (storage.bucket(bucket).getFiles({ prefix }) as any) as Promise<[File[]]>,\n      {\n        name: 'neo_backup_list_files',\n      },\n    );\n    const times = [...new Set(fileNames.map((file) => extractTime(prefix, file)))];\n    // tslint:disable-next-line no-array-mutation\n    times.sort();\n\n    const deleteTimes = times.slice(0, -keepBackupCount);\n    await monitor.captureSpanLog<Promise<void[]>>(\n      async () =>\n        Promise.all(\n          deleteTimes.map(async (deleteTime) =>\n            storage.bucket(bucket).deleteFiles({ prefix: [prefix, `${deleteTime}`].join('/') }),\n          ),\n        ),\n      { name: 'neo_backup_delete_old' },\n    );\n  }\n\n  private async getLatestTime(): Promise<{\n    readonly time: number | undefined;\n    readonly files: ReadonlyArray<File>;\n  }> {\n    const { bucket, prefix } = this.options;\n\n    const storage = await this.getStorage();\n    // tslint:disable-next-line no-any no-void-expression no-use-of-empty-return-value\n    const [files] = (await (storage.bucket(bucket).getFiles({ prefix }) as any)) as [File[]];\n\n    const metadataTimes = files\n      .filter((file) => path.basename(file.name) === METADATA_NAME)\n      .map((file) => extractTime(prefix, file));\n    // tslint:disable-next-line no-array-mutation\n    metadataTimes.sort();\n\n    const time = metadataTimes[metadataTimes.length - 1] as number | undefined;\n\n    return { time, files };\n  }\n\n  private async getStorage() {\n    const storage = await import('@google-cloud/storage');\n\n    // tslint:disable-next-line no-any\n    return new storage.Storage({ projectId: this.options.projectID });\n  }\n}\n"]}