mirror of
https://github.com/google-gemini/gemini-cli
synced 2026-04-21 13:37:17 +00:00
116 lines
3.4 KiB
TypeScript
116 lines
3.4 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2026 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
import type { CommandModule } from 'yargs';
|
|
import { debugLogger, type SkillDefinition } from '@google/gemini-cli-core';
|
|
import { getErrorMessage } from '../../utils/errors.js';
|
|
import { exitCli } from '../utils.js';
|
|
import { installSkill } from '../../utils/skillUtils.js';
|
|
import chalk from 'chalk';
|
|
import {
|
|
requestConsentNonInteractive,
|
|
skillsConsentString,
|
|
} from '../../config/extensions/consent.js';
|
|
|
|
interface InstallArgs {
|
|
source: string;
|
|
scope?: 'user' | 'workspace';
|
|
path?: string;
|
|
consent?: boolean;
|
|
}
|
|
|
|
export async function handleInstall(args: InstallArgs) {
|
|
try {
|
|
const { source, consent } = args;
|
|
const scope = args.scope ?? 'user';
|
|
const subpath = args.path;
|
|
|
|
const requestConsent = async (
|
|
skills: SkillDefinition[],
|
|
targetDir: string,
|
|
) => {
|
|
if (consent) {
|
|
debugLogger.log('You have consented to the following:');
|
|
debugLogger.log(await skillsConsentString(skills, source, targetDir));
|
|
return true;
|
|
}
|
|
return requestConsentNonInteractive(
|
|
await skillsConsentString(skills, source, targetDir),
|
|
);
|
|
};
|
|
|
|
const installedSkills = await installSkill(
|
|
source,
|
|
scope,
|
|
subpath,
|
|
(msg) => {
|
|
debugLogger.log(msg);
|
|
},
|
|
requestConsent,
|
|
);
|
|
|
|
for (const skill of installedSkills) {
|
|
debugLogger.log(
|
|
chalk.green(
|
|
`Successfully installed skill: ${chalk.bold(skill.name)} (scope: ${scope}, location: ${skill.location})`,
|
|
),
|
|
);
|
|
}
|
|
} catch (error) {
|
|
debugLogger.error(getErrorMessage(error));
|
|
await exitCli(1);
|
|
}
|
|
}
|
|
|
|
export const installCommand: CommandModule = {
|
|
command: 'install <source> [--scope] [--path]',
|
|
describe:
|
|
'Installs an agent skill from a git repository URL or a local path.',
|
|
builder: (yargs) =>
|
|
yargs
|
|
.positional('source', {
|
|
describe:
|
|
'The git repository URL or local path of the skill to install.',
|
|
type: 'string',
|
|
demandOption: true,
|
|
})
|
|
.option('scope', {
|
|
describe:
|
|
'The scope to install the skill into. Defaults to "user" (global).',
|
|
choices: ['user', 'workspace'],
|
|
default: 'user',
|
|
})
|
|
.option('path', {
|
|
describe:
|
|
'Sub-path within the repository to install from (only used for git repository sources).',
|
|
type: 'string',
|
|
})
|
|
.option('consent', {
|
|
describe:
|
|
'Acknowledge the security risks of installing a skill and skip the confirmation prompt.',
|
|
type: 'boolean',
|
|
default: false,
|
|
})
|
|
.check((argv) => {
|
|
if (!argv.source) {
|
|
throw new Error('The source argument must be provided.');
|
|
}
|
|
return true;
|
|
}),
|
|
handler: async (argv) => {
|
|
await handleInstall({
|
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-type-assertion
|
|
source: argv['source'] as string,
|
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-type-assertion
|
|
scope: argv['scope'] as 'user' | 'workspace',
|
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-type-assertion
|
|
path: argv['path'] as string | undefined,
|
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-type-assertion
|
|
consent: argv['consent'] as boolean | undefined,
|
|
});
|
|
await exitCli();
|
|
},
|
|
};
|