@optima-chat/ads-cli 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude/CLAUDE.md +279 -0
- package/dist/commands/account/check.d.ts +6 -0
- package/dist/commands/account/check.d.ts.map +1 -0
- package/dist/commands/account/check.js +179 -0
- package/dist/commands/account/check.js.map +1 -0
- package/dist/commands/account/create.d.ts +6 -0
- package/dist/commands/account/create.d.ts.map +1 -0
- package/dist/commands/account/create.js +172 -0
- package/dist/commands/account/create.js.map +1 -0
- package/dist/commands/account/index.d.ts +6 -0
- package/dist/commands/account/index.d.ts.map +1 -0
- package/dist/commands/account/index.js +11 -0
- package/dist/commands/account/index.js.map +1 -0
- package/dist/commands/ad/create.d.ts +6 -0
- package/dist/commands/ad/create.d.ts.map +1 -0
- package/dist/commands/ad/create.js +77 -0
- package/dist/commands/ad/create.js.map +1 -0
- package/dist/commands/ad/delete.d.ts +6 -0
- package/dist/commands/ad/delete.d.ts.map +1 -0
- package/dist/commands/ad/delete.js +50 -0
- package/dist/commands/ad/delete.js.map +1 -0
- package/dist/commands/ad/index.d.ts +6 -0
- package/dist/commands/ad/index.d.ts.map +1 -0
- package/dist/commands/ad/index.js +17 -0
- package/dist/commands/ad/index.js.map +1 -0
- package/dist/commands/ad/info.d.ts +6 -0
- package/dist/commands/ad/info.d.ts.map +1 -0
- package/dist/commands/ad/info.js +90 -0
- package/dist/commands/ad/info.js.map +1 -0
- package/dist/commands/ad/list.d.ts +6 -0
- package/dist/commands/ad/list.d.ts.map +1 -0
- package/dist/commands/ad/list.js +94 -0
- package/dist/commands/ad/list.js.map +1 -0
- package/dist/commands/ad/update.d.ts +6 -0
- package/dist/commands/ad/update.d.ts.map +1 -0
- package/dist/commands/ad/update.js +42 -0
- package/dist/commands/ad/update.js.map +1 -0
- package/dist/commands/ad-group/create.d.ts +6 -0
- package/dist/commands/ad-group/create.d.ts.map +1 -0
- package/dist/commands/ad-group/create.js +59 -0
- package/dist/commands/ad-group/create.js.map +1 -0
- package/dist/commands/ad-group/delete.d.ts +6 -0
- package/dist/commands/ad-group/delete.d.ts.map +1 -0
- package/dist/commands/ad-group/delete.js +49 -0
- package/dist/commands/ad-group/delete.js.map +1 -0
- package/dist/commands/ad-group/index.d.ts +6 -0
- package/dist/commands/ad-group/index.d.ts.map +1 -0
- package/dist/commands/ad-group/index.js +17 -0
- package/dist/commands/ad-group/index.js.map +1 -0
- package/dist/commands/ad-group/info.d.ts +6 -0
- package/dist/commands/ad-group/info.d.ts.map +1 -0
- package/dist/commands/ad-group/info.js +59 -0
- package/dist/commands/ad-group/info.js.map +1 -0
- package/dist/commands/ad-group/list.d.ts +6 -0
- package/dist/commands/ad-group/list.d.ts.map +1 -0
- package/dist/commands/ad-group/list.js +83 -0
- package/dist/commands/ad-group/list.js.map +1 -0
- package/dist/commands/ad-group/update.d.ts +6 -0
- package/dist/commands/ad-group/update.d.ts.map +1 -0
- package/dist/commands/ad-group/update.js +56 -0
- package/dist/commands/ad-group/update.js.map +1 -0
- package/dist/commands/auth/index.d.ts +6 -0
- package/dist/commands/auth/index.d.ts.map +1 -0
- package/dist/commands/auth/index.js +13 -0
- package/dist/commands/auth/index.js.map +1 -0
- package/dist/commands/auth/login.d.ts +10 -0
- package/dist/commands/auth/login.d.ts.map +1 -0
- package/dist/commands/auth/login.js +154 -0
- package/dist/commands/auth/login.js.map +1 -0
- package/dist/commands/auth/logout.d.ts +6 -0
- package/dist/commands/auth/logout.d.ts.map +1 -0
- package/dist/commands/auth/logout.js +28 -0
- package/dist/commands/auth/logout.js.map +1 -0
- package/dist/commands/auth/status.d.ts +6 -0
- package/dist/commands/auth/status.d.ts.map +1 -0
- package/dist/commands/auth/status.js +47 -0
- package/dist/commands/auth/status.js.map +1 -0
- package/dist/commands/campaign/create.d.ts +6 -0
- package/dist/commands/campaign/create.d.ts.map +1 -0
- package/dist/commands/campaign/create.js +58 -0
- package/dist/commands/campaign/create.js.map +1 -0
- package/dist/commands/campaign/delete.d.ts +6 -0
- package/dist/commands/campaign/delete.d.ts.map +1 -0
- package/dist/commands/campaign/delete.js +49 -0
- package/dist/commands/campaign/delete.js.map +1 -0
- package/dist/commands/campaign/index.d.ts +6 -0
- package/dist/commands/campaign/index.d.ts.map +1 -0
- package/dist/commands/campaign/index.js +17 -0
- package/dist/commands/campaign/index.js.map +1 -0
- package/dist/commands/campaign/info.d.ts +6 -0
- package/dist/commands/campaign/info.d.ts.map +1 -0
- package/dist/commands/campaign/info.js +57 -0
- package/dist/commands/campaign/info.js.map +1 -0
- package/dist/commands/campaign/list.d.ts +6 -0
- package/dist/commands/campaign/list.d.ts.map +1 -0
- package/dist/commands/campaign/list.js +71 -0
- package/dist/commands/campaign/list.js.map +1 -0
- package/dist/commands/campaign/update.d.ts +6 -0
- package/dist/commands/campaign/update.d.ts.map +1 -0
- package/dist/commands/campaign/update.js +56 -0
- package/dist/commands/campaign/update.js.map +1 -0
- package/dist/commands/config.d.ts +6 -0
- package/dist/commands/config.d.ts.map +1 -0
- package/dist/commands/config.js +104 -0
- package/dist/commands/config.js.map +1 -0
- package/dist/commands/keyword/add.d.ts +6 -0
- package/dist/commands/keyword/add.d.ts.map +1 -0
- package/dist/commands/keyword/add.js +47 -0
- package/dist/commands/keyword/add.js.map +1 -0
- package/dist/commands/keyword/delete.d.ts +6 -0
- package/dist/commands/keyword/delete.d.ts.map +1 -0
- package/dist/commands/keyword/delete.js +50 -0
- package/dist/commands/keyword/delete.js.map +1 -0
- package/dist/commands/keyword/index.d.ts +6 -0
- package/dist/commands/keyword/index.d.ts.map +1 -0
- package/dist/commands/keyword/index.js +15 -0
- package/dist/commands/keyword/index.js.map +1 -0
- package/dist/commands/keyword/list.d.ts +6 -0
- package/dist/commands/keyword/list.d.ts.map +1 -0
- package/dist/commands/keyword/list.js +77 -0
- package/dist/commands/keyword/list.js.map +1 -0
- package/dist/commands/keyword/update.d.ts +6 -0
- package/dist/commands/keyword/update.d.ts.map +1 -0
- package/dist/commands/keyword/update.js +51 -0
- package/dist/commands/keyword/update.js.map +1 -0
- package/dist/commands/query.d.ts +6 -0
- package/dist/commands/query.d.ts.map +1 -0
- package/dist/commands/query.js +71 -0
- package/dist/commands/query.js.map +1 -0
- package/dist/config.d.ts +32 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +10 -0
- package/dist/config.js.map +1 -0
- package/dist/index.d.ts +7 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +75 -0
- package/dist/index.js.map +1 -0
- package/dist/lib/api-client.d.ts +100 -0
- package/dist/lib/api-client.d.ts.map +1 -0
- package/dist/lib/api-client.js +186 -0
- package/dist/lib/api-client.js.map +1 -0
- package/dist/lib/callback-server.d.ts +32 -0
- package/dist/lib/callback-server.d.ts.map +1 -0
- package/dist/lib/callback-server.js +132 -0
- package/dist/lib/callback-server.js.map +1 -0
- package/dist/lib/google-ads-client.d.ts +177 -0
- package/dist/lib/google-ads-client.d.ts.map +1 -0
- package/dist/lib/google-ads-client.js +826 -0
- package/dist/lib/google-ads-client.js.map +1 -0
- package/dist/lib/oauth2-manager.d.ts +27 -0
- package/dist/lib/oauth2-manager.d.ts.map +1 -0
- package/dist/lib/oauth2-manager.js +123 -0
- package/dist/lib/oauth2-manager.js.map +1 -0
- package/dist/lib/token-store.d.ts +43 -0
- package/dist/lib/token-store.d.ts.map +1 -0
- package/dist/lib/token-store.js +103 -0
- package/dist/lib/token-store.js.map +1 -0
- package/dist/utils/config.d.ts +9 -0
- package/dist/utils/config.d.ts.map +1 -0
- package/dist/utils/config.js +34 -0
- package/dist/utils/config.js.map +1 -0
- package/dist/utils/customer-id.d.ts +10 -0
- package/dist/utils/customer-id.d.ts.map +1 -0
- package/dist/utils/customer-id.js +29 -0
- package/dist/utils/customer-id.js.map +1 -0
- package/dist/utils/env-updater.d.ts +8 -0
- package/dist/utils/env-updater.d.ts.map +1 -0
- package/dist/utils/env-updater.js +45 -0
- package/dist/utils/env-updater.js.map +1 -0
- package/dist/utils/errors.d.ts +32 -0
- package/dist/utils/errors.d.ts.map +1 -0
- package/dist/utils/errors.js +68 -0
- package/dist/utils/errors.js.map +1 -0
- package/dist/utils/logger.d.ts +10 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +25 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/optima-config.d.ts +9 -0
- package/dist/utils/optima-config.d.ts.map +1 -0
- package/dist/utils/optima-config.js +83 -0
- package/dist/utils/optima-config.js.map +1 -0
- package/package.json +67 -0
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Keyword Add Command - 添加关键词
|
|
3
|
+
*/
|
|
4
|
+
import { Command } from 'commander';
|
|
5
|
+
import chalk from 'chalk';
|
|
6
|
+
import ora from 'ora';
|
|
7
|
+
import { GoogleAdsClient } from '../../lib/google-ads-client.js';
|
|
8
|
+
import { handleError } from '../../utils/errors.js';
|
|
9
|
+
export const addCommand = new Command('add')
|
|
10
|
+
.description('添加关键词')
|
|
11
|
+
.requiredOption('-c, --customer-id <id>', '客户账号 ID')
|
|
12
|
+
.requiredOption('--ad-group-id <id>', '广告组 ID')
|
|
13
|
+
.requiredOption('-k, --keywords <keywords>', '关键词(逗号分隔)')
|
|
14
|
+
.option('--match-type <type>', '匹配类型 (BROAD, PHRASE, EXACT)', 'BROAD')
|
|
15
|
+
.option('--json', '以 JSON 格式输出')
|
|
16
|
+
.action(async (options) => {
|
|
17
|
+
const spinner = ora('添加关键词...').start();
|
|
18
|
+
try {
|
|
19
|
+
const client = new GoogleAdsClient();
|
|
20
|
+
// 解析关键词
|
|
21
|
+
const keywordTexts = options.keywords.split(',').map((k) => k.trim());
|
|
22
|
+
const keywords = keywordTexts.map((text) => ({
|
|
23
|
+
text,
|
|
24
|
+
match_type: options.matchType,
|
|
25
|
+
}));
|
|
26
|
+
const result = await client.addKeywords(options.customerId, options.adGroupId, keywords);
|
|
27
|
+
spinner.succeed(`成功添加 ${keywordTexts.length} 个关键词`);
|
|
28
|
+
if (options.json) {
|
|
29
|
+
console.log(JSON.stringify(result, null, 2));
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
console.log(chalk.green(`\n✅ 已添加 ${keywordTexts.length} 个关键词`));
|
|
33
|
+
console.log(chalk.gray('匹配类型:'), chalk.white(options.matchType));
|
|
34
|
+
console.log(chalk.gray('\n关键词列表:'));
|
|
35
|
+
keywordTexts.forEach((text, index) => {
|
|
36
|
+
console.log(chalk.white(` ${index + 1}. ${text}`));
|
|
37
|
+
});
|
|
38
|
+
console.log(chalk.gray('\n💡 查看关键词:'));
|
|
39
|
+
console.log(chalk.cyan(` google-ads keyword list -c ${options.customerId} --ad-group-id ${options.adGroupId}`));
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
catch (error) {
|
|
43
|
+
spinner.fail('添加失败');
|
|
44
|
+
handleError(error);
|
|
45
|
+
}
|
|
46
|
+
});
|
|
47
|
+
//# sourceMappingURL=add.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"add.js","sourceRoot":"","sources":["../../../src/commands/keyword/add.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,GAAG,MAAM,KAAK,CAAC;AACtB,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AACjE,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAEpD,MAAM,CAAC,MAAM,UAAU,GAAG,IAAI,OAAO,CAAC,KAAK,CAAC;KACzC,WAAW,CAAC,OAAO,CAAC;KACpB,cAAc,CAAC,wBAAwB,EAAE,SAAS,CAAC;KACnD,cAAc,CAAC,oBAAoB,EAAE,QAAQ,CAAC;KAC9C,cAAc,CAAC,2BAA2B,EAAE,WAAW,CAAC;KACxD,MAAM,CAAC,qBAAqB,EAAE,6BAA6B,EAAE,OAAO,CAAC;KACrE,MAAM,CAAC,QAAQ,EAAE,aAAa,CAAC;KAC/B,MAAM,CAAC,KAAK,EAAE,OAAO,EAAE,EAAE;IACxB,MAAM,OAAO,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,KAAK,EAAE,CAAC;IAExC,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;QAErC,QAAQ;QACR,MAAM,YAAY,GAAG,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;QAC9E,MAAM,QAAQ,GAAG,YAAY,CAAC,GAAG,CAAC,CAAC,IAAY,EAAE,EAAE,CAAC,CAAC;YACnD,IAAI;YACJ,UAAU,EAAE,OAAO,CAAC,SAAS;SAC9B,CAAC,CAAC,CAAC;QAEJ,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,WAAW,CACrC,OAAO,CAAC,UAAU,EAClB,OAAO,CAAC,SAAS,EACjB,QAAQ,CACT,CAAC;QAEF,OAAO,CAAC,OAAO,CAAC,QAAQ,YAAY,CAAC,MAAM,OAAO,CAAC,CAAC;QAEpD,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC/C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,WAAW,YAAY,CAAC,MAAM,OAAO,CAAC,CAAC,CAAC;YAChE,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC;YACjE,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;YACpC,YAAY,CAAC,OAAO,CAAC,CAAC,IAAY,EAAE,KAAa,EAAE,EAAE;gBACnD,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,KAAK,KAAK,GAAG,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,CAAC;YACtD,CAAC,CAAC,CAAC;YAEH,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;YACvC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,iCAAiC,OAAO,CAAC,UAAU,kBAAkB,OAAO,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;QACpH,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACrB,WAAW,CAAC,KAAK,CAAC,CAAC;IACrB,CAAC;AACH,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"delete.d.ts","sourceRoot":"","sources":["../../../src/commands/keyword/delete.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAQpC,eAAO,MAAM,aAAa,SA8CtB,CAAC"}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Keyword Delete Command - 删除关键词
|
|
3
|
+
*/
|
|
4
|
+
import { Command } from 'commander';
|
|
5
|
+
import chalk from 'chalk';
|
|
6
|
+
import ora from 'ora';
|
|
7
|
+
import inquirer from 'inquirer';
|
|
8
|
+
import { GoogleAdsClient } from '../../lib/google-ads-client.js';
|
|
9
|
+
import { handleError } from '../../utils/errors.js';
|
|
10
|
+
import { getCustomerId } from '../../utils/customer-id.js';
|
|
11
|
+
export const deleteCommand = new Command('delete')
|
|
12
|
+
.description('删除关键词')
|
|
13
|
+
.requiredOption('--ad-group-id <id>', '广告组 ID')
|
|
14
|
+
.requiredOption('--criterion-id <id>', '关键词 Criterion ID')
|
|
15
|
+
.option('-y, --yes', '跳过确认')
|
|
16
|
+
.option('--json', '以 JSON 格式输出')
|
|
17
|
+
.action(async (options) => {
|
|
18
|
+
try {
|
|
19
|
+
const customerId = getCustomerId();
|
|
20
|
+
// 确认删除
|
|
21
|
+
if (!options.yes) {
|
|
22
|
+
const answer = await inquirer.prompt([
|
|
23
|
+
{
|
|
24
|
+
type: 'confirm',
|
|
25
|
+
name: 'confirm',
|
|
26
|
+
message: `确定要删除关键词 ${options.criterionId} 吗?`,
|
|
27
|
+
default: false,
|
|
28
|
+
},
|
|
29
|
+
]);
|
|
30
|
+
if (!answer.confirm) {
|
|
31
|
+
console.log(chalk.yellow('已取消'));
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
const spinner = ora('删除关键词...').start();
|
|
36
|
+
const client = new GoogleAdsClient();
|
|
37
|
+
const result = await client.deleteKeyword(customerId, options.adGroupId, options.criterionId);
|
|
38
|
+
spinner.succeed('关键词已删除');
|
|
39
|
+
if (options.json) {
|
|
40
|
+
console.log(JSON.stringify(result, null, 2));
|
|
41
|
+
}
|
|
42
|
+
else {
|
|
43
|
+
console.log(chalk.green(`\n✅ 关键词 ${options.criterionId} 已删除`));
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
catch (error) {
|
|
47
|
+
handleError(error);
|
|
48
|
+
}
|
|
49
|
+
});
|
|
50
|
+
//# sourceMappingURL=delete.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"delete.js","sourceRoot":"","sources":["../../../src/commands/keyword/delete.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,GAAG,MAAM,KAAK,CAAC;AACtB,OAAO,QAAQ,MAAM,UAAU,CAAC;AAChC,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AACjE,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AACpD,OAAO,EAAE,aAAa,EAAE,MAAM,4BAA4B,CAAC;AAE3D,MAAM,CAAC,MAAM,aAAa,GAAG,IAAI,OAAO,CAAC,QAAQ,CAAC;KAC/C,WAAW,CAAC,OAAO,CAAC;KACpB,cAAc,CAAC,oBAAoB,EAAE,QAAQ,CAAC;KAC9C,cAAc,CAAC,qBAAqB,EAAE,kBAAkB,CAAC;KACzD,MAAM,CAAC,WAAW,EAAE,MAAM,CAAC;KAC3B,MAAM,CAAC,QAAQ,EAAE,aAAa,CAAC;KAC/B,MAAM,CAAC,KAAK,EAAE,OAAO,EAAE,EAAE;IACxB,IAAI,CAAC;QACH,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;QAEnC,OAAO;QACP,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC;YACjB,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,MAAM,CAAC;gBACnC;oBACE,IAAI,EAAE,SAAS;oBACf,IAAI,EAAE,SAAS;oBACf,OAAO,EAAE,YAAY,OAAO,CAAC,WAAW,KAAK;oBAC7C,OAAO,EAAE,KAAK;iBACf;aACF,CAAC,CAAC;YAEH,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;gBACpB,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;gBACjC,OAAO;YACT,CAAC;QACH,CAAC;QAED,MAAM,OAAO,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,KAAK,EAAE,CAAC;QAExC,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;QACrC,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,aAAa,CACvC,UAAU,EACV,OAAO,CAAC,SAAS,EACjB,OAAO,CAAC,WAAW,CACpB,CAAC;QAEF,OAAO,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QAE1B,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC/C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,WAAW,OAAO,CAAC,WAAW,MAAM,CAAC,CAAC,CAAC;QACjE,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,WAAW,CAAC,KAAK,CAAC,CAAC;IACrB,CAAC;AACH,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/commands/keyword/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAMpC,eAAO,MAAM,cAAc,SAKC,CAAC"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Keyword Commands - 关键词管理
|
|
3
|
+
*/
|
|
4
|
+
import { Command } from 'commander';
|
|
5
|
+
import { listCommand } from './list.js';
|
|
6
|
+
import { addCommand } from './add.js';
|
|
7
|
+
import { updateCommand } from './update.js';
|
|
8
|
+
import { deleteCommand } from './delete.js';
|
|
9
|
+
export const keywordCommand = new Command('keyword')
|
|
10
|
+
.description('关键词管理')
|
|
11
|
+
.addCommand(listCommand)
|
|
12
|
+
.addCommand(addCommand)
|
|
13
|
+
.addCommand(updateCommand)
|
|
14
|
+
.addCommand(deleteCommand);
|
|
15
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/commands/keyword/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AACxC,OAAO,EAAE,UAAU,EAAE,MAAM,UAAU,CAAC;AACtC,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAC5C,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAE5C,MAAM,CAAC,MAAM,cAAc,GAAG,IAAI,OAAO,CAAC,SAAS,CAAC;KACjD,WAAW,CAAC,OAAO,CAAC;KACpB,UAAU,CAAC,WAAW,CAAC;KACvB,UAAU,CAAC,UAAU,CAAC;KACtB,UAAU,CAAC,aAAa,CAAC;KACzB,UAAU,CAAC,aAAa,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"list.d.ts","sourceRoot":"","sources":["../../../src/commands/keyword/list.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AASpC,eAAO,MAAM,WAAW,SAuEpB,CAAC"}
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Keyword List Command - 列出关键词
|
|
3
|
+
*/
|
|
4
|
+
import { Command } from 'commander';
|
|
5
|
+
import chalk from 'chalk';
|
|
6
|
+
import Table from 'cli-table3';
|
|
7
|
+
import ora from 'ora';
|
|
8
|
+
import { GoogleAdsClient } from '../../lib/google-ads-client.js';
|
|
9
|
+
import { handleError } from '../../utils/errors.js';
|
|
10
|
+
import { warn } from '../../utils/logger.js';
|
|
11
|
+
import { getCustomerId } from '../../utils/customer-id.js';
|
|
12
|
+
export const listCommand = new Command('list')
|
|
13
|
+
.description('列出关键词')
|
|
14
|
+
.option('--campaign-id <id>', '按广告系列过滤')
|
|
15
|
+
.option('--status <status>', '按状态过滤 (ENABLED/PAUSED/REMOVED)')
|
|
16
|
+
.option('--limit <number>', '限制返回数量', '100')
|
|
17
|
+
.option('--json', '以 JSON 格式输出')
|
|
18
|
+
.action(async (options) => {
|
|
19
|
+
try {
|
|
20
|
+
const customerId = getCustomerId();
|
|
21
|
+
const client = new GoogleAdsClient();
|
|
22
|
+
const spinner = ora('正在获取关键词列表...').start();
|
|
23
|
+
// 获取关键词列表
|
|
24
|
+
const keywords = await client.listKeywords(customerId, options.campaignId, {
|
|
25
|
+
status: options.status,
|
|
26
|
+
limit: parseInt(options.limit),
|
|
27
|
+
});
|
|
28
|
+
spinner.stop();
|
|
29
|
+
if (keywords.length === 0) {
|
|
30
|
+
warn('未找到关键词');
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
// 输出结果
|
|
34
|
+
if (options.json) {
|
|
35
|
+
console.log(JSON.stringify(keywords, null, 2));
|
|
36
|
+
}
|
|
37
|
+
else {
|
|
38
|
+
const table = new Table({
|
|
39
|
+
head: [
|
|
40
|
+
chalk.cyan('关键词'),
|
|
41
|
+
chalk.cyan('匹配类型'),
|
|
42
|
+
chalk.cyan('状态'),
|
|
43
|
+
chalk.cyan('质量得分'),
|
|
44
|
+
chalk.cyan('展示量'),
|
|
45
|
+
chalk.cyan('点击量'),
|
|
46
|
+
chalk.cyan('费用'),
|
|
47
|
+
chalk.cyan('点击率'),
|
|
48
|
+
],
|
|
49
|
+
colWidths: [25, 12, 10, 10, 12, 10, 12, 10],
|
|
50
|
+
});
|
|
51
|
+
keywords.forEach((keyword) => {
|
|
52
|
+
const cost = keyword.metrics?.cost_micros
|
|
53
|
+
? (keyword.metrics.cost_micros / 1000000).toFixed(2)
|
|
54
|
+
: '0.00';
|
|
55
|
+
const ctr = keyword.metrics?.ctr
|
|
56
|
+
? (keyword.metrics.ctr * 100).toFixed(2) + '%'
|
|
57
|
+
: '0.00%';
|
|
58
|
+
table.push([
|
|
59
|
+
keyword.ad_group_criterion?.keyword?.text || 'N/A',
|
|
60
|
+
keyword.ad_group_criterion?.keyword?.match_type || 'N/A',
|
|
61
|
+
keyword.ad_group_criterion?.status || 'N/A',
|
|
62
|
+
keyword.ad_group_criterion?.quality_info?.quality_score || 'N/A',
|
|
63
|
+
keyword.metrics?.impressions || 0,
|
|
64
|
+
keyword.metrics?.clicks || 0,
|
|
65
|
+
cost,
|
|
66
|
+
ctr,
|
|
67
|
+
]);
|
|
68
|
+
});
|
|
69
|
+
console.log('\n' + table.toString() + '\n');
|
|
70
|
+
console.log(chalk.gray(`共 ${keywords.length} 个关键词\n`));
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
catch (error) {
|
|
74
|
+
handleError(error);
|
|
75
|
+
}
|
|
76
|
+
});
|
|
77
|
+
//# sourceMappingURL=list.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"list.js","sourceRoot":"","sources":["../../../src/commands/keyword/list.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"update.d.ts","sourceRoot":"","sources":["../../../src/commands/keyword/update.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAOpC,eAAO,MAAM,aAAa,SA0CtB,CAAC"}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Keyword Update Command - 更新关键词
|
|
3
|
+
*/
|
|
4
|
+
import { Command } from 'commander';
|
|
5
|
+
import chalk from 'chalk';
|
|
6
|
+
import ora from 'ora';
|
|
7
|
+
import { GoogleAdsClient } from '../../lib/google-ads-client.js';
|
|
8
|
+
import { handleError } from '../../utils/errors.js';
|
|
9
|
+
import { getCustomerId } from '../../utils/customer-id.js';
|
|
10
|
+
export const updateCommand = new Command('update')
|
|
11
|
+
.description('更新关键词')
|
|
12
|
+
.requiredOption('--ad-group-id <id>', '广告组 ID')
|
|
13
|
+
.requiredOption('--criterion-id <id>', '关键词条件 ID')
|
|
14
|
+
.option('--status <status>', '状态 (ENABLED, PAUSED)')
|
|
15
|
+
.option('--bid <amount>', '每次点击出价(美元)')
|
|
16
|
+
.option('--json', '以 JSON 格式输出')
|
|
17
|
+
.action(async (options) => {
|
|
18
|
+
if (!options.status && !options.bid) {
|
|
19
|
+
console.log(chalk.red('请至少指定一个要更新的字段: --status 或 --bid'));
|
|
20
|
+
return;
|
|
21
|
+
}
|
|
22
|
+
const spinner = ora('更新关键词...').start();
|
|
23
|
+
try {
|
|
24
|
+
const customerId = getCustomerId();
|
|
25
|
+
const client = new GoogleAdsClient();
|
|
26
|
+
const updateData = {};
|
|
27
|
+
if (options.status) {
|
|
28
|
+
updateData.status = options.status;
|
|
29
|
+
}
|
|
30
|
+
if (options.bid) {
|
|
31
|
+
updateData.cpcBidMicros = Math.round(parseFloat(options.bid) * 1000000);
|
|
32
|
+
}
|
|
33
|
+
const result = await client.updateKeyword(customerId, options.adGroupId, options.criterionId, updateData);
|
|
34
|
+
spinner.succeed('关键词已更新');
|
|
35
|
+
if (options.json) {
|
|
36
|
+
console.log(JSON.stringify(result, null, 2));
|
|
37
|
+
}
|
|
38
|
+
else {
|
|
39
|
+
console.log(chalk.green(`\n✅ 关键词 ${options.criterionId} 已更新`));
|
|
40
|
+
if (options.status)
|
|
41
|
+
console.log(chalk.gray(`状态: ${options.status}`));
|
|
42
|
+
if (options.bid)
|
|
43
|
+
console.log(chalk.gray(`出价: $${options.bid}`));
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
catch (error) {
|
|
47
|
+
spinner.fail('更新失败');
|
|
48
|
+
handleError(error);
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
//# sourceMappingURL=update.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"update.js","sourceRoot":"","sources":["../../../src/commands/keyword/update.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,GAAG,MAAM,KAAK,CAAC;AACtB,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AACjE,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AACpD,OAAO,EAAE,aAAa,EAAE,MAAM,4BAA4B,CAAC;AAE3D,MAAM,CAAC,MAAM,aAAa,GAAG,IAAI,OAAO,CAAC,QAAQ,CAAC;KAC/C,WAAW,CAAC,OAAO,CAAC;KACpB,cAAc,CAAC,oBAAoB,EAAE,QAAQ,CAAC;KAC9C,cAAc,CAAC,qBAAqB,EAAE,UAAU,CAAC;KACjD,MAAM,CAAC,mBAAmB,EAAE,sBAAsB,CAAC;KACnD,MAAM,CAAC,gBAAgB,EAAE,YAAY,CAAC;KACtC,MAAM,CAAC,QAAQ,EAAE,aAAa,CAAC;KAC/B,MAAM,CAAC,KAAK,EAAE,OAAO,EAAE,EAAE;IACxB,IAAI,CAAC,OAAO,CAAC,MAAM,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC;QACpC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,iCAAiC,CAAC,CAAC,CAAC;QAC1D,OAAO;IACT,CAAC;IAED,MAAM,OAAO,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,KAAK,EAAE,CAAC;IAExC,IAAI,CAAC;QACH,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;QACnC,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;QAErC,MAAM,UAAU,GAAQ,EAAE,CAAC;QAC3B,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;YACnB,UAAU,CAAC,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;QACrC,CAAC;QACD,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC;YAChB,UAAU,CAAC,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,CAAC;QAC1E,CAAC;QAED,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,aAAa,CAAC,UAAU,EAAE,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,WAAW,EAAE,UAAU,CAAC,CAAC;QAE1G,OAAO,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QAE1B,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC/C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,WAAW,OAAO,CAAC,WAAW,MAAM,CAAC,CAAC,CAAC;YAC/D,IAAI,OAAO,CAAC,MAAM;gBAAE,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YACrE,IAAI,OAAO,CAAC,GAAG;gBAAE,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC;QAClE,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACrB,WAAW,CAAC,KAAK,CAAC,CAAC;IACrB,CAAC;AACH,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"query.d.ts","sourceRoot":"","sources":["../../src/commands/query.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AASpC,eAAO,MAAM,YAAY,SA+DrB,CAAC"}
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Query Command - 执行 GAQL 查询
|
|
3
|
+
*/
|
|
4
|
+
import { Command } from 'commander';
|
|
5
|
+
import chalk from 'chalk';
|
|
6
|
+
import ora from 'ora';
|
|
7
|
+
import { readFileSync } from 'fs';
|
|
8
|
+
import { GoogleAdsClient } from '../lib/google-ads-client.js';
|
|
9
|
+
import { handleError } from '../utils/errors.js';
|
|
10
|
+
import { warn } from '../utils/logger.js';
|
|
11
|
+
import { getCustomerId } from '../utils/customer-id.js';
|
|
12
|
+
export const queryCommand = new Command('query')
|
|
13
|
+
.description('执行 GAQL 查询')
|
|
14
|
+
.option('-c, --customer-id <id>', '客户账号 ID(可选,默认从配置读取)')
|
|
15
|
+
.option('-q, --query <query>', 'GAQL 查询语句')
|
|
16
|
+
.option('-f, --file <file>', '从文件读取 GAQL 查询')
|
|
17
|
+
.option('--json', '以 JSON 格式输出(默认)')
|
|
18
|
+
.option('--pretty', '美化 JSON 输出')
|
|
19
|
+
.action(async (options) => {
|
|
20
|
+
try {
|
|
21
|
+
// 获取查询语句
|
|
22
|
+
let query;
|
|
23
|
+
if (options.query) {
|
|
24
|
+
query = options.query;
|
|
25
|
+
}
|
|
26
|
+
else if (options.file) {
|
|
27
|
+
query = readFileSync(options.file, 'utf-8');
|
|
28
|
+
}
|
|
29
|
+
else {
|
|
30
|
+
warn('请提供查询语句(使用 -q 或 -f 参数)');
|
|
31
|
+
console.log();
|
|
32
|
+
console.log(chalk.gray('示例:'));
|
|
33
|
+
console.log(chalk.cyan(' google-ads query -c 1234567890 -q "SELECT campaign.id, campaign.name FROM campaign"'));
|
|
34
|
+
console.log(chalk.cyan(' google-ads query -c 1234567890 -f query.gaql'));
|
|
35
|
+
process.exit(1);
|
|
36
|
+
}
|
|
37
|
+
// 清理查询语句
|
|
38
|
+
query = query.trim();
|
|
39
|
+
if (!query) {
|
|
40
|
+
warn('查询语句不能为空');
|
|
41
|
+
process.exit(1);
|
|
42
|
+
}
|
|
43
|
+
const client = new GoogleAdsClient();
|
|
44
|
+
const customerId = options.customerId || getCustomerId();
|
|
45
|
+
const spinner = ora('正在执行查询...').start();
|
|
46
|
+
// 执行查询
|
|
47
|
+
const results = await client.query(customerId, query);
|
|
48
|
+
spinner.stop();
|
|
49
|
+
// 输出结果
|
|
50
|
+
if (results.length === 0) {
|
|
51
|
+
warn('查询未返回任何结果');
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
// 总是输出 JSON 格式(LLM 友好)
|
|
55
|
+
if (options.pretty) {
|
|
56
|
+
console.log(JSON.stringify(results, null, 2));
|
|
57
|
+
}
|
|
58
|
+
else {
|
|
59
|
+
console.log(JSON.stringify(results));
|
|
60
|
+
}
|
|
61
|
+
// 显示统计信息(仅当 pretty 模式时)
|
|
62
|
+
if (options.pretty) {
|
|
63
|
+
console.log();
|
|
64
|
+
console.log(chalk.gray(`共返回 ${results.length} 条记录`));
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
catch (error) {
|
|
68
|
+
handleError(error);
|
|
69
|
+
}
|
|
70
|
+
});
|
|
71
|
+
//# sourceMappingURL=query.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"query.js","sourceRoot":"","sources":["../../src/commands/query.ts"],"names":[],"mappings":"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"}
|
package/dist/config.d.ts
ADDED
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 全局配置类型定义
|
|
3
|
+
*/
|
|
4
|
+
/**
|
|
5
|
+
* Google Ads CLI 配置
|
|
6
|
+
*/
|
|
7
|
+
export interface GoogleAdsConfig {
|
|
8
|
+
developerToken: string;
|
|
9
|
+
clientId: string;
|
|
10
|
+
clientSecret: string;
|
|
11
|
+
loginCustomerId?: string;
|
|
12
|
+
defaultCustomerId?: string;
|
|
13
|
+
outputFormat: 'table' | 'json' | 'csv' | 'jsonl';
|
|
14
|
+
colorOutput: boolean;
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* OAuth2 Token
|
|
18
|
+
*/
|
|
19
|
+
export interface OAuth2Token {
|
|
20
|
+
access_token: string;
|
|
21
|
+
refresh_token: string;
|
|
22
|
+
expires_at: number;
|
|
23
|
+
scope: string;
|
|
24
|
+
token_type: string;
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* 配置文件路径
|
|
28
|
+
*/
|
|
29
|
+
export declare const CONFIG_DIR = ".config/google-ads-cli";
|
|
30
|
+
export declare const CONFIG_FILE = "config.json";
|
|
31
|
+
export declare const CREDENTIALS_FILE = "credentials.json";
|
|
32
|
+
//# sourceMappingURL=config.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH;;GAEG;AACH,MAAM,WAAW,eAAe;IAE9B,cAAc,EAAE,MAAM,CAAC;IACvB,QAAQ,EAAE,MAAM,CAAC;IACjB,YAAY,EAAE,MAAM,CAAC;IAGrB,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAG3B,YAAY,EAAE,OAAO,GAAG,MAAM,GAAG,KAAK,GAAG,OAAO,CAAC;IACjD,WAAW,EAAE,OAAO,CAAC;CACtB;AAED;;GAEG;AACH,MAAM,WAAW,WAAW;IAC1B,YAAY,EAAE,MAAM,CAAC;IACrB,aAAa,EAAE,MAAM,CAAC;IACtB,UAAU,EAAE,MAAM,CAAC;IACnB,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,MAAM,CAAC;CACpB;AAED;;GAEG;AACH,eAAO,MAAM,UAAU,2BAA2B,CAAC;AACnD,eAAO,MAAM,WAAW,gBAAgB,CAAC;AACzC,eAAO,MAAM,gBAAgB,qBAAqB,CAAC"}
|
package/dist/config.js
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAAA;;GAEG;AA+BH;;GAEG;AACH,MAAM,CAAC,MAAM,UAAU,GAAG,wBAAwB,CAAC;AACnD,MAAM,CAAC,MAAM,WAAW,GAAG,aAAa,CAAC;AACzC,MAAM,CAAC,MAAM,gBAAgB,GAAG,kBAAkB,CAAC"}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAEA;;;GAGG;AAGH,OAAO,eAAe,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* Google Ads CLI - Main Entry Point
|
|
4
|
+
* 用自然语言管理 Google Ads 广告投放 - 专为 Claude Code 设计
|
|
5
|
+
*/
|
|
6
|
+
// 加载 .env 文件中的环境变量(静默模式)
|
|
7
|
+
import 'dotenv/config';
|
|
8
|
+
import { Command } from 'commander';
|
|
9
|
+
import chalk from 'chalk';
|
|
10
|
+
import { readFileSync } from 'fs';
|
|
11
|
+
import { fileURLToPath } from 'url';
|
|
12
|
+
import { dirname, join } from 'path';
|
|
13
|
+
import { authCommand } from './commands/auth/index.js';
|
|
14
|
+
import { accountCommand } from './commands/account/index.js';
|
|
15
|
+
import { campaignCommand } from './commands/campaign/index.js';
|
|
16
|
+
import { adGroupCommand } from './commands/ad-group/index.js';
|
|
17
|
+
import { adCommand } from './commands/ad/index.js';
|
|
18
|
+
import { keywordCommand } from './commands/keyword/index.js';
|
|
19
|
+
import { queryCommand } from './commands/query.js';
|
|
20
|
+
import { configCommand } from './commands/config.js';
|
|
21
|
+
import { fetchOptimaConfig } from './utils/optima-config.js';
|
|
22
|
+
// 获取版本号
|
|
23
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
24
|
+
const __dirname = dirname(__filename);
|
|
25
|
+
const packageJson = JSON.parse(readFileSync(join(__dirname, '../package.json'), 'utf-8'));
|
|
26
|
+
const VERSION = packageJson.version;
|
|
27
|
+
// 创建主程序
|
|
28
|
+
const program = new Command();
|
|
29
|
+
program
|
|
30
|
+
.name('google-ads')
|
|
31
|
+
.description('用自然语言管理 Google Ads 广告投放 - 专为 Claude Code 设计')
|
|
32
|
+
.version(VERSION);
|
|
33
|
+
// 注册命令
|
|
34
|
+
program.addCommand(authCommand);
|
|
35
|
+
program.addCommand(accountCommand);
|
|
36
|
+
program.addCommand(campaignCommand);
|
|
37
|
+
program.addCommand(adGroupCommand);
|
|
38
|
+
program.addCommand(adCommand);
|
|
39
|
+
program.addCommand(keywordCommand);
|
|
40
|
+
program.addCommand(queryCommand);
|
|
41
|
+
program.addCommand(configCommand);
|
|
42
|
+
// 默认动作(显示欢迎信息)
|
|
43
|
+
program.action(() => {
|
|
44
|
+
console.log(chalk.cyan.bold(`\n✨ Google Ads CLI v${VERSION}\n`));
|
|
45
|
+
console.log(chalk.white('管理您的 Google Ads 广告投放'));
|
|
46
|
+
console.log(chalk.gray('专为 Claude Code 设计的 CLI 工具\n'));
|
|
47
|
+
console.log(chalk.green('🚀 首次使用:'));
|
|
48
|
+
console.log(chalk.white(' 步骤 1: 配置 Agency 凭据'));
|
|
49
|
+
console.log(chalk.gray(' • 复制 .env.example 为 .env'));
|
|
50
|
+
console.log(chalk.gray(' • 填入 Agency 提供的凭据(Developer Token, OAuth2, MCC ID)'));
|
|
51
|
+
console.log(chalk.gray(' • 运行: ') + chalk.cyan('google-ads auth login\n'));
|
|
52
|
+
console.log(chalk.white(' 步骤 2: 创建您的账号'));
|
|
53
|
+
console.log(chalk.gray(' • 运行: ') + chalk.cyan('google-ads account create --email <your-email> --name <your-name>'));
|
|
54
|
+
console.log(chalk.gray(' • 检查邮箱接受邀请'));
|
|
55
|
+
console.log(chalk.gray(' • 设置账单信息\n'));
|
|
56
|
+
console.log(chalk.green('📋 日常使用:'));
|
|
57
|
+
console.log(chalk.gray(' • 检查账号状态: ') + chalk.cyan('google-ads account check'));
|
|
58
|
+
console.log(chalk.gray(' • 查看广告系列: ') + chalk.cyan('google-ads campaign list'));
|
|
59
|
+
console.log(chalk.gray(' • 创建广告系列: ') + chalk.cyan('google-ads campaign create -n "广告系列名称" -b 10'));
|
|
60
|
+
console.log(chalk.gray(' • 查看所有命令: ') + chalk.cyan('google-ads --help') + chalk.gray('\n'));
|
|
61
|
+
console.log(chalk.white('⚙️ 配置管理:'));
|
|
62
|
+
console.log(chalk.gray(' • 查看当前配置: ') + chalk.cyan('google-ads config show'));
|
|
63
|
+
console.log(chalk.gray(' • 重置配置: ') + chalk.cyan('google-ads config reset\n'));
|
|
64
|
+
console.log(chalk.white('📖 文档:'));
|
|
65
|
+
console.log(chalk.gray(' README: https://github.com/Optima-Chat/google-ads-cli#readme\n'));
|
|
66
|
+
});
|
|
67
|
+
// 启动 CLI
|
|
68
|
+
async function main() {
|
|
69
|
+
// 从 commerce-backend 获取配置(如果有 OPTIMA_TOKEN)
|
|
70
|
+
await fetchOptimaConfig();
|
|
71
|
+
// 解析命令行参数
|
|
72
|
+
program.parse();
|
|
73
|
+
}
|
|
74
|
+
main();
|
|
75
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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"}
|