Skip to content

Commit

Permalink
[kbn/dev-utils] pull in extract() helper (#106277)
Browse files Browse the repository at this point in the history
Co-authored-by: spalger <spalger@users.noreply.github.com>
  • Loading branch information
Spencer and spalger authored Jul 20, 2021
1 parent 20a9470 commit ec160d5
Show file tree
Hide file tree
Showing 11 changed files with 150 additions and 151 deletions.
9 changes: 7 additions & 2 deletions packages/kbn-dev-utils/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ NPM_MODULE_EXTRA_FILES = [

SRC_DEPS = [
"//packages/kbn-expect",
"//packages/kbn-std",
"//packages/kbn-utils",
"@npm//@babel/core",
"@npm//axios",
Expand All @@ -60,10 +61,12 @@ SRC_DEPS = [
"@npm//moment",
"@npm//normalize-path",
"@npm//rxjs",
"@npm//tar",
"@npm//tree-kill",
"@npm//tslib",
"@npm//typescript",
"@npm//vinyl"
"@npm//vinyl",
"@npm//yauzl"
]

TYPES_DEPS = [
Expand All @@ -76,8 +79,10 @@ TYPES_DEPS = [
"@npm//@types/node",
"@npm//@types/normalize-path",
"@npm//@types/react",
"@npm//@types/tar",
"@npm//@types/testing-library__jest-dom",
"@npm//@types/vinyl"
"@npm//@types/vinyl",
"@npm//@types/yauzl"
]

DEPS = SRC_DEPS + TYPES_DEPS
Expand Down
120 changes: 120 additions & 0 deletions packages/kbn-dev-utils/src/extract.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,120 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0 and the Server Side Public License, v 1; you may not use this file except
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/

import Fs from 'fs/promises';
import { createWriteStream } from 'fs';
import Path from 'path';
import { pipeline } from 'stream';
import { promisify } from 'util';

import { lastValueFrom } from '@kbn/std';
import Tar from 'tar';
import Yauzl, { ZipFile, Entry } from 'yauzl';
import * as Rx from 'rxjs';
import { map, mergeMap, takeUntil } from 'rxjs/operators';

const asyncPipeline = promisify(pipeline);

interface Options {
/**
* Path to the archive to extract, .tar, .tar.gz, and .zip archives are supported
*/
archivePath: string;

/**
* Directory where the contents of the archive will be written. Existing files in that
* directory will be overwritten. If the directory doesn't exist it will be created.
*/
targetDir: string;

/**
* Number of path segments to strip form paths in the archive, like --strip-components from tar
*/
stripComponents?: number;
}

/**
* Extract tar and zip archives using a single function, supporting stripComponents
* for both archive types, only tested with familiar archives we create so might not
* support some weird exotic zip features we don't use in our own snapshot/build tooling
*/
export async function extract({ archivePath, targetDir, stripComponents = 0 }: Options) {
await Fs.mkdir(targetDir, { recursive: true });

if (archivePath.endsWith('.tar') || archivePath.endsWith('.tar.gz')) {
return await Tar.x({
file: archivePath,
cwd: targetDir,
stripComponents,
});
}

if (!archivePath.endsWith('.zip')) {
throw new Error('unsupported archive type');
}

// zip mode
const zipFile = await new Promise<ZipFile>((resolve, reject) => {
Yauzl.open(archivePath, { lazyEntries: true }, (error, _zipFile) => {
if (error || !_zipFile) {
reject(error || new Error('no zipfile provided by yauzl'));
} else {
resolve(_zipFile);
}
});
});

// bound version of zipFile.openReadStream which returns an observable, because of type defs the readStream
// result is technically optional (thanks callbacks)
const openReadStream$ = Rx.bindNodeCallback(zipFile.openReadStream.bind(zipFile));

const close$ = Rx.fromEvent(zipFile, 'close');
const error$ = Rx.fromEvent<Error>(zipFile, 'error').pipe(
takeUntil(close$),
map((error) => {
throw error;
})
);

const entry$ = Rx.fromEvent<Entry>(zipFile, 'entry').pipe(
takeUntil(close$),
mergeMap((entry) => {
const entryPath = entry.fileName.split(/\/|\\/).slice(stripComponents).join(Path.sep);
const fileName = Path.resolve(targetDir, entryPath);

// detect directories
if (entry.fileName.endsWith('/')) {
return Rx.defer(async () => {
// ensure the directory exists
await Fs.mkdir(fileName, { recursive: true });
// tell yauzl to read the next entry
zipFile.readEntry();
});
}

// file entry
return openReadStream$(entry).pipe(
mergeMap(async (readStream) => {
if (!readStream) {
throw new Error('no readstream provided by yauzl');
}

// write the file contents to disk
await asyncPipeline(readStream, createWriteStream(fileName));
// tell yauzl to read the next entry
zipFile.readEntry();
})
);
})
);

// trigger the initial 'entry' event, happens async so the event will be delivered after the observable is subscribed
zipFile.readEntry();

await lastValueFrom(Rx.merge(entry$, error$));
}
1 change: 1 addition & 0 deletions packages/kbn-dev-utils/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,3 +31,4 @@ export * from './plugin_list';
export * from './plugins';
export * from './streams';
export * from './babel';
export * from './extract';
3 changes: 2 additions & 1 deletion packages/kbn-dev-utils/src/run/help.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@

import Path from 'path';

import chalk from 'chalk';
import 'core-js/features/string/repeat';
import dedent from 'dedent';

Expand Down Expand Up @@ -116,7 +117,7 @@ export function getHelpForAllCommands({
: '';

return [
dedent(command.usage || '') || command.name,
chalk.bold.whiteBright.bgBlack(` ${dedent(command.usage || '') || command.name} `),
` ${indent(dedent(command.description || 'Runs a dev task'), 2)}`,
...([indent(options, 2)] || []),
].join('\n');
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,12 @@
* Side Public License, v 1.
*/

export function createAnyInstanceSerializer(Class: Function, name?: string) {
export function createAnyInstanceSerializer(
Class: Function,
name?: string | ((instance: any) => string)
) {
return {
test: (v: any) => v instanceof Class,
serialize: () => `<${name ?? Class.name}>`,
serialize: (v: any) => `<${typeof name === 'function' ? name(v) : name ?? Class.name}>`,
};
}
2 changes: 0 additions & 2 deletions packages/kbn-es/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -39,9 +39,7 @@ DEPS = [
"@npm//glob",
"@npm//node-fetch",
"@npm//simple-git",
"@npm//tar-fs",
"@npm//tree-kill",
"@npm//yauzl",
"@npm//zlib"
]

Expand Down
18 changes: 8 additions & 10 deletions packages/kbn-es/src/cluster.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,18 +13,12 @@ const chalk = require('chalk');
const path = require('path');
const { downloadSnapshot, installSnapshot, installSource, installArchive } = require('./install');
const { ES_BIN } = require('./paths');
const {
log: defaultLog,
parseEsLog,
extractConfigFiles,
decompress,
NativeRealm,
} = require('./utils');
const { log: defaultLog, parseEsLog, extractConfigFiles, NativeRealm } = require('./utils');
const { createCliError } = require('./errors');
const { promisify } = require('util');
const treeKillAsync = promisify(require('tree-kill'));
const { parseSettings, SettingsFilter } = require('./settings');
const { CA_CERT_PATH, ES_P12_PATH, ES_P12_PASSWORD } = require('@kbn/dev-utils');
const { CA_CERT_PATH, ES_P12_PATH, ES_P12_PASSWORD, extract } = require('@kbn/dev-utils');
const readFile = util.promisify(fs.readFile);

// listen to data on stream until map returns anything but undefined
Expand Down Expand Up @@ -144,13 +138,17 @@ exports.Cluster = class Cluster {
this._log.info(chalk.bold(`Extracting data directory`));
this._log.indent(4);

// decompress excludes the root directory as that is how our archives are
// stripComponents=1 excludes the root directory as that is how our archives are
// structured. This works in our favor as we can explicitly extract into the data dir
const extractPath = path.resolve(installPath, extractDirName);
this._log.info(`Data archive: ${archivePath}`);
this._log.info(`Extract path: ${extractPath}`);

await decompress(archivePath, extractPath);
await extract({
archivePath,
targetDir: extractPath,
stripComponents: 1,
});

this._log.indent(-4);
}
Expand Down
9 changes: 7 additions & 2 deletions packages/kbn-es/src/install/archive.js
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,8 @@ const chalk = require('chalk');
const execa = require('execa');
const del = require('del');
const url = require('url');
const { log: defaultLog, decompress } = require('../utils');
const { extract } = require('@kbn/dev-utils');
const { log: defaultLog } = require('../utils');
const { BASE_PATH, ES_CONFIG, ES_KEYSTORE_BIN } = require('../paths');
const { Artifact } = require('../artifact');
const { parseSettings, SettingsFilter } = require('../settings');
Expand Down Expand Up @@ -50,7 +51,11 @@ exports.installArchive = async function installArchive(archive, options = {}) {
}

log.info('extracting %s', chalk.bold(dest));
await decompress(dest, installPath);
await extract({
archivePath: dest,
targetDir: installPath,
stripComponents: 1,
});
log.info('extracted to %s', chalk.bold(installPath));

const tmpdir = path.resolve(installPath, 'ES_TMPDIR');
Expand Down
86 changes: 0 additions & 86 deletions packages/kbn-es/src/utils/decompress.js

This file was deleted.

45 changes: 0 additions & 45 deletions packages/kbn-es/src/utils/decompress.test.js

This file was deleted.

Loading

0 comments on commit ec160d5

Please sign in to comment.