Skip to content
This repository has been archived by the owner on Mar 31, 2024. It is now read-only.

Commit

Permalink
[kbn/es-archiver] move to a package (elastic#72318)
Browse files Browse the repository at this point in the history
Co-authored-by: spalger <[email protected]>
Co-authored-by: Elastic Machine <[email protected]>
# Conflicts:
#	.github/CODEOWNERS
#	src/dev/code_coverage/ingest_coverage/team_assignment/ingestion_pipeline_painless.json
  • Loading branch information
Spencer authored and spalger committed Jul 22, 2020
1 parent de4abba commit ca003fb
Show file tree
Hide file tree
Showing 57 changed files with 334 additions and 252 deletions.
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -295,6 +295,7 @@
"@elastic/makelogs": "^6.0.0",
"@kbn/dev-utils": "1.0.0",
"@kbn/es": "1.0.0",
"@kbn/es-archiver": "1.0.0",
"@kbn/eslint-import-resolver-kibana": "2.0.0",
"@kbn/eslint-plugin-eslint": "1.0.0",
"@kbn/expect": "1.0.0",
Expand Down
4 changes: 1 addition & 3 deletions packages/kbn-dev-utils/src/run/run_with_commands.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -61,9 +61,7 @@ it('extends the context using extendContext()', async () => {

expect(context.flags).toMatchInlineSnapshot(`
Object {
"_": Array [
"foo",
],
"_": Array [],
"debug": false,
"help": false,
"quiet": false,
Expand Down
9 changes: 8 additions & 1 deletion packages/kbn-dev-utils/src/run/run_with_commands.ts
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,13 @@ export class RunWithCommands<T> {

const commandFlagOptions = mergeFlagOptions(this.options.globalFlags, command.flags);
const commandFlags = getFlags(process.argv.slice(2), commandFlagOptions);
// strip command name plus "help" if we're actually executing the fake "help" command
if (isHelpCommand) {
commandFlags._.splice(0, 2);
} else {
commandFlags._.splice(0, 1);
}

const commandHelp = getCommandLevelHelp({
usage: this.options.usage,
globalFlagHelp: this.options.globalFlags?.help,
Expand All @@ -115,7 +122,7 @@ export class RunWithCommands<T> {
log,
flags: commandFlags,
procRunner,
addCleanupTask: cleanup.add,
addCleanupTask: cleanup.add.bind(cleanup),
};

const extendedContext = {
Expand Down
17 changes: 17 additions & 0 deletions packages/kbn-es-archiver/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
{
"name": "@kbn/es-archiver",
"version": "1.0.0",
"license": "Apache-2.0",
"main": "target/index.js",
"scripts": {
"kbn:bootstrap": "tsc",
"kbn:watch": "tsc --watch"
},
"dependencies": {
"@kbn/dev-utils": "1.0.0",
"elasticsearch": "^16.7.0"
},
"devDependencies": {
"@types/elasticsearch": "^5.0.33"
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ import { promisify } from 'util';
import globby from 'globby';
import { ToolingLog } from '@kbn/dev-utils';

import { createPromiseFromStreams } from '../../legacy/utils';
import { createPromiseFromStreams } from '../lib/streams';

const unlinkAsync = promisify(Fs.unlink);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ import { Readable } from 'stream';
import { ToolingLog, KbnClient } from '@kbn/dev-utils';
import { Client } from 'elasticsearch';

import { createPromiseFromStreams, concatStreamProviders } from '../../legacy/utils';
import { createPromiseFromStreams, concatStreamProviders } from '../lib/streams';

import {
isGzip,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ import { Readable, Writable } from 'stream';
import { fromNode } from 'bluebird';
import { ToolingLog } from '@kbn/dev-utils';

import { createPromiseFromStreams } from '../../legacy/utils';
import { createPromiseFromStreams } from '../lib/streams';
import {
prioritizeMappings,
readDirectory,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ import { Readable, Writable } from 'stream';
import { Client } from 'elasticsearch';
import { ToolingLog } from '@kbn/dev-utils';

import { createListStream, createPromiseFromStreams } from '../../legacy/utils';
import { createListStream, createPromiseFromStreams } from '../lib/streams';
import {
createStats,
createGenerateIndexRecordsStream,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ import { Readable, Writable } from 'stream';
import { Client } from 'elasticsearch';
import { ToolingLog, KbnClient } from '@kbn/dev-utils';

import { createPromiseFromStreams } from '../../legacy/utils';
import { createPromiseFromStreams } from '../lib/streams';
import {
isGzip,
createStats,
Expand Down
244 changes: 244 additions & 0 deletions packages/kbn-es-archiver/src/cli.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,244 @@
/*
* Licensed to Elasticsearch B.V. under one or more contributor
* license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright
* ownership. Elasticsearch B.V. licenses this file to you 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.
*/

/** ***********************************************************
*
* Run `node scripts/es_archiver --help` for usage information
*
*************************************************************/

import Path from 'path';
import Url from 'url';
import readline from 'readline';

import { RunWithCommands, createFlagError } from '@kbn/dev-utils';
import { readConfigFile } from '@kbn/test';
import legacyElasticsearch from 'elasticsearch';

import { EsArchiver } from './es_archiver';

const resolveConfigPath = (v: string) => Path.resolve(process.cwd(), v);
const defaultConfigPath = resolveConfigPath('test/functional/config.js');

export function runCli() {
new RunWithCommands({
description: 'CLI to manage archiving/restoring data in elasticsearch',
globalFlags: {
string: ['es-url', 'kibana-url', 'dir', 'config'],
help: `
--config path to an FTR config file that sets --es-url, --kibana-url, and --dir
default: ${defaultConfigPath}
--es-url url for Elasticsearch, prefer the --config flag
--kibana-url url for Kibana, prefer the --config flag
--dir where arechives are stored, prefer the --config flag
`,
},
async extendContext({ log, flags, addCleanupTask }) {
const configPath = flags.config || defaultConfigPath;
if (typeof configPath !== 'string') {
throw createFlagError('--config must be a string');
}
const config = await readConfigFile(log, Path.resolve(configPath));

let esUrl = flags['es-url'];
if (esUrl && typeof esUrl !== 'string') {
throw createFlagError('--es-url must be a string');
}
if (!esUrl && config) {
esUrl = Url.format(config.get('servers.elasticsearch'));
}
if (!esUrl) {
throw createFlagError('--es-url or --config must be defined');
}

let kibanaUrl = flags['kibana-url'];
if (kibanaUrl && typeof kibanaUrl !== 'string') {
throw createFlagError('--kibana-url must be a string');
}
if (!kibanaUrl && config) {
kibanaUrl = Url.format(config.get('servers.kibana'));
}
if (!kibanaUrl) {
throw createFlagError('--kibana-url or --config must be defined');
}

let dir = flags.dir;
if (dir && typeof dir !== 'string') {
throw createFlagError('--dir must be a string');
}
if (!dir && config) {
dir = Path.resolve(config.get('esArchiver.directory'));
}
if (!dir) {
throw createFlagError('--dir or --config must be defined');
}

const client = new legacyElasticsearch.Client({
host: esUrl,
log: flags.verbose ? 'trace' : [],
});
addCleanupTask(() => client.close());

const esArchiver = new EsArchiver({
log,
client,
dataDir: dir,
kibanaUrl,
});

return {
esArchiver,
};
},
})
.command({
name: 'save',
usage: 'save [name] [...indices]',
description: `
archive the [indices ...] into the --dir with [name]
Example:
Save all [logstash-*] indices from http://localhost:9200 to [snapshots/my_test_data] directory
WARNING: If the [my_test_data] snapshot exists it will be deleted!
$ node scripts/es_archiver save my_test_data logstash-* --dir snapshots
`,
flags: {
boolean: ['raw'],
help: `
--raw don't gzip the archives
`,
},
async run({ flags, esArchiver }) {
const [name, ...indices] = flags._;
if (!name) {
throw createFlagError('missing [name] argument');
}
if (!indices.length) {
throw createFlagError('missing [...indices] arguments');
}

const raw = flags.raw;
if (typeof raw !== 'boolean') {
throw createFlagError('--raw does not take a value');
}

await esArchiver.save(name, indices, { raw });
},
})
.command({
name: 'load',
usage: 'load [name]',
description: `
load the archive in --dir with [name]
Example:
Load the [my_test_data] snapshot from the archive directory and elasticsearch instance defined
in the [test/functional/config.js] config file
WARNING: If the indices exist already they will be deleted!
$ node scripts/es_archiver load my_test_data --config test/functional/config.js
`,
flags: {
boolean: ['use-create'],
help: `
--use-create use create instead of index for loading documents
`,
},
async run({ flags, esArchiver }) {
const [name] = flags._;
if (!name) {
throw createFlagError('missing [name] argument');
}
if (flags._.length > 1) {
throw createFlagError(`unknown extra arguments: [${flags._.slice(1).join(', ')}]`);
}

const useCreate = flags['use-create'];
if (typeof useCreate !== 'boolean') {
throw createFlagError('--use-create does not take a value');
}

await esArchiver.load(name, { useCreate });
},
})
.command({
name: 'unload',
usage: 'unload [name]',
description: 'remove indices created by the archive in --dir with [name]',
async run({ flags, esArchiver }) {
const [name] = flags._;
if (!name) {
throw createFlagError('missing [name] argument');
}
if (flags._.length > 1) {
throw createFlagError(`unknown extra arguments: [${flags._.slice(1).join(', ')}]`);
}

await esArchiver.unload(name);
},
})
.command({
name: 'edit',
usage: 'edit [prefix]',
description:
'extract the archives under the prefix, wait for edits to be completed, and then recompress the archives',
async run({ flags, esArchiver }) {
const [prefix] = flags._;
if (!prefix) {
throw createFlagError('missing [prefix] argument');
}
if (flags._.length > 1) {
throw createFlagError(`unknown extra arguments: [${flags._.slice(1).join(', ')}]`);
}

await esArchiver.edit(prefix, async () => {
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout,
});

await new Promise((resolveInput) => {
rl.question(`Press enter when you're done`, () => {
rl.close();
resolveInput();
});
});
});
},
})
.command({
name: 'empty-kibana-index',
description:
'[internal] Delete any Kibana indices, and initialize the Kibana index as Kibana would do on startup.',
async run({ esArchiver }) {
await esArchiver.emptyKibanaIndex();
},
})
.command({
name: 'rebuild-all',
description: '[internal] read and write all archives in --dir to remove any inconsistencies',
async run({ esArchiver }) {
await esArchiver.rebuildAll();
},
})
.execute();
}
Original file line number Diff line number Diff line change
Expand Up @@ -18,3 +18,4 @@
*/

export { EsArchiver } from './es_archiver';
export * from './cli';
Original file line number Diff line number Diff line change
Expand Up @@ -22,11 +22,7 @@ import { createGunzip } from 'zlib';

import expect from '@kbn/expect';

import {
createListStream,
createPromiseFromStreams,
createConcatStream,
} from '../../../../legacy/utils';
import { createListStream, createPromiseFromStreams, createConcatStream } from '../../streams';

import { createFormatArchiveStreams } from '../format';

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,11 +22,7 @@ import { createGzip } from 'zlib';

import expect from '@kbn/expect';

import {
createConcatStream,
createListStream,
createPromiseFromStreams,
} from '../../../../legacy/utils';
import { createConcatStream, createListStream, createPromiseFromStreams } from '../../streams';

import { createParseArchiveStreams } from '../parse';

Expand Down Expand Up @@ -109,7 +105,7 @@ describe('esArchiver createParseArchiveStreams', () => {
Buffer.from('{"a": 2}\n\n'),
]),
...createParseArchiveStreams({ gzip: false }),
createConcatStream(),
createConcatStream([]),
] as [Readable, ...Writable[]]);
throw new Error('should have failed');
} catch (err) {
Expand Down Expand Up @@ -172,7 +168,7 @@ describe('esArchiver createParseArchiveStreams', () => {
await createPromiseFromStreams([
createListStream([Buffer.from('{"a": 1}')]),
...createParseArchiveStreams({ gzip: true }),
createConcatStream(),
createConcatStream([]),
] as [Readable, ...Writable[]]);
throw new Error('should have failed');
} catch (err) {
Expand Down
Loading

0 comments on commit ca003fb

Please sign in to comment.