aidevops 2.52.1 → 2.53.2

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.
Files changed (334) hide show
  1. package/README.md +1 -1
  2. package/VERSION +1 -1
  3. package/aidevops.sh +15 -9
  4. package/package.json +4 -4
  5. package/scripts/npm-postinstall.js +6 -7
  6. package/setup.sh +1 -1
  7. package/templates/deploy-templates.sh +144 -0
  8. package/templates/home/.agent/README.md +33 -0
  9. package/templates/home/AGENTS.md +96 -0
  10. package/templates/home/git/.agent/README.md +48 -0
  11. package/templates/home/git/AGENTS.md +97 -0
  12. package/templates/standard-functions.sh +179 -0
  13. package/templates/wordpress-performance-workflow.md +217 -0
  14. package/.agent/AGENTS.md +0 -614
  15. package/.agent/accounts.md +0 -65
  16. package/.agent/aidevops/add-new-mcp-to-aidevops.md +0 -456
  17. package/.agent/aidevops/api-integrations.md +0 -335
  18. package/.agent/aidevops/architecture.md +0 -510
  19. package/.agent/aidevops/configs.md +0 -274
  20. package/.agent/aidevops/docs.md +0 -244
  21. package/.agent/aidevops/extension.md +0 -311
  22. package/.agent/aidevops/mcp-integrations.md +0 -340
  23. package/.agent/aidevops/mcp-troubleshooting.md +0 -162
  24. package/.agent/aidevops/memory-patterns.md +0 -172
  25. package/.agent/aidevops/providers.md +0 -217
  26. package/.agent/aidevops/recommendations.md +0 -321
  27. package/.agent/aidevops/requirements.md +0 -301
  28. package/.agent/aidevops/resources.md +0 -214
  29. package/.agent/aidevops/security-requirements.md +0 -174
  30. package/.agent/aidevops/security.md +0 -350
  31. package/.agent/aidevops/service-links.md +0 -400
  32. package/.agent/aidevops/services.md +0 -357
  33. package/.agent/aidevops/setup.md +0 -153
  34. package/.agent/aidevops/troubleshooting.md +0 -389
  35. package/.agent/aidevops.md +0 -124
  36. package/.agent/build-plus.md +0 -244
  37. package/.agent/content/guidelines.md +0 -109
  38. package/.agent/content.md +0 -87
  39. package/.agent/health.md +0 -59
  40. package/.agent/legal.md +0 -59
  41. package/.agent/loop-state/full-loop.local.md +0 -16
  42. package/.agent/loop-state/ralph-loop.local.md +0 -10
  43. package/.agent/marketing.md +0 -440
  44. package/.agent/memory/README.md +0 -260
  45. package/.agent/onboarding.md +0 -796
  46. package/.agent/plan-plus.md +0 -245
  47. package/.agent/research.md +0 -100
  48. package/.agent/sales.md +0 -333
  49. package/.agent/scripts/101domains-helper.sh +0 -701
  50. package/.agent/scripts/add-missing-returns.sh +0 -140
  51. package/.agent/scripts/agent-browser-helper.sh +0 -311
  52. package/.agent/scripts/agno-setup.sh +0 -712
  53. package/.agent/scripts/ahrefs-mcp-wrapper.js +0 -168
  54. package/.agent/scripts/aidevops-update-check.sh +0 -71
  55. package/.agent/scripts/ampcode-cli.sh +0 -522
  56. package/.agent/scripts/auto-version-bump.sh +0 -156
  57. package/.agent/scripts/autogen-helper.sh +0 -512
  58. package/.agent/scripts/beads-sync-helper.sh +0 -596
  59. package/.agent/scripts/closte-helper.sh +0 -5
  60. package/.agent/scripts/cloudron-helper.sh +0 -321
  61. package/.agent/scripts/codacy-cli-chunked.sh +0 -581
  62. package/.agent/scripts/codacy-cli.sh +0 -442
  63. package/.agent/scripts/code-audit-helper.sh +0 -5
  64. package/.agent/scripts/coderabbit-cli.sh +0 -417
  65. package/.agent/scripts/coderabbit-pro-analysis.sh +0 -238
  66. package/.agent/scripts/commands/code-simplifier.md +0 -86
  67. package/.agent/scripts/commands/full-loop.md +0 -246
  68. package/.agent/scripts/commands/postflight-loop.md +0 -103
  69. package/.agent/scripts/commands/recall.md +0 -182
  70. package/.agent/scripts/commands/remember.md +0 -132
  71. package/.agent/scripts/commands/save-todo.md +0 -175
  72. package/.agent/scripts/commands/session-review.md +0 -154
  73. package/.agent/scripts/comprehensive-quality-fix.sh +0 -106
  74. package/.agent/scripts/context-builder-helper.sh +0 -522
  75. package/.agent/scripts/coolify-cli-helper.sh +0 -674
  76. package/.agent/scripts/coolify-helper.sh +0 -380
  77. package/.agent/scripts/crawl4ai-examples.sh +0 -401
  78. package/.agent/scripts/crawl4ai-helper.sh +0 -1078
  79. package/.agent/scripts/crewai-helper.sh +0 -681
  80. package/.agent/scripts/dev-browser-helper.sh +0 -513
  81. package/.agent/scripts/dns-helper.sh +0 -396
  82. package/.agent/scripts/domain-research-helper.sh +0 -917
  83. package/.agent/scripts/dspy-helper.sh +0 -285
  84. package/.agent/scripts/dspyground-helper.sh +0 -291
  85. package/.agent/scripts/eeat-score-helper.sh +0 -1242
  86. package/.agent/scripts/efficient-return-fix.sh +0 -92
  87. package/.agent/scripts/extract-opencode-prompts.sh +0 -128
  88. package/.agent/scripts/find-missing-returns.sh +0 -113
  89. package/.agent/scripts/fix-auth-headers.sh +0 -104
  90. package/.agent/scripts/fix-common-strings.sh +0 -254
  91. package/.agent/scripts/fix-content-type.sh +0 -100
  92. package/.agent/scripts/fix-error-messages.sh +0 -130
  93. package/.agent/scripts/fix-misplaced-returns.sh +0 -74
  94. package/.agent/scripts/fix-remaining-literals.sh +0 -152
  95. package/.agent/scripts/fix-return-statements.sh +0 -41
  96. package/.agent/scripts/fix-s131-default-cases.sh +0 -249
  97. package/.agent/scripts/fix-sc2155-simple.sh +0 -102
  98. package/.agent/scripts/fix-shellcheck-critical.sh +0 -187
  99. package/.agent/scripts/fix-string-literals.sh +0 -273
  100. package/.agent/scripts/full-loop-helper.sh +0 -773
  101. package/.agent/scripts/generate-opencode-agents.sh +0 -497
  102. package/.agent/scripts/generate-opencode-commands.sh +0 -1629
  103. package/.agent/scripts/generate-skills.sh +0 -366
  104. package/.agent/scripts/git-platforms-helper.sh +0 -640
  105. package/.agent/scripts/gitea-cli-helper.sh +0 -743
  106. package/.agent/scripts/github-cli-helper.sh +0 -702
  107. package/.agent/scripts/gitlab-cli-helper.sh +0 -682
  108. package/.agent/scripts/gsc-add-user-helper.sh +0 -325
  109. package/.agent/scripts/gsc-sitemap-helper.sh +0 -678
  110. package/.agent/scripts/hetzner-helper.sh +0 -485
  111. package/.agent/scripts/hostinger-helper.sh +0 -229
  112. package/.agent/scripts/keyword-research-helper.sh +0 -1815
  113. package/.agent/scripts/langflow-helper.sh +0 -544
  114. package/.agent/scripts/linkedin-automation.py +0 -241
  115. package/.agent/scripts/linter-manager.sh +0 -599
  116. package/.agent/scripts/linters-local.sh +0 -434
  117. package/.agent/scripts/list-keys-helper.sh +0 -488
  118. package/.agent/scripts/local-browser-automation.py +0 -339
  119. package/.agent/scripts/localhost-helper.sh +0 -744
  120. package/.agent/scripts/loop-common.sh +0 -806
  121. package/.agent/scripts/mainwp-helper.sh +0 -728
  122. package/.agent/scripts/markdown-formatter.sh +0 -338
  123. package/.agent/scripts/markdown-lint-fix.sh +0 -311
  124. package/.agent/scripts/mass-fix-returns.sh +0 -58
  125. package/.agent/scripts/mcp-diagnose.sh +0 -167
  126. package/.agent/scripts/mcp-inspector-helper.sh +0 -449
  127. package/.agent/scripts/memory-helper.sh +0 -650
  128. package/.agent/scripts/monitor-code-review.sh +0 -255
  129. package/.agent/scripts/onboarding-helper.sh +0 -706
  130. package/.agent/scripts/opencode-github-setup-helper.sh +0 -797
  131. package/.agent/scripts/opencode-test-helper.sh +0 -213
  132. package/.agent/scripts/pagespeed-helper.sh +0 -464
  133. package/.agent/scripts/pandoc-helper.sh +0 -362
  134. package/.agent/scripts/postflight-check.sh +0 -555
  135. package/.agent/scripts/pre-commit-hook.sh +0 -259
  136. package/.agent/scripts/pre-edit-check.sh +0 -169
  137. package/.agent/scripts/qlty-cli.sh +0 -356
  138. package/.agent/scripts/quality-cli-manager.sh +0 -525
  139. package/.agent/scripts/quality-feedback-helper.sh +0 -462
  140. package/.agent/scripts/quality-fix.sh +0 -263
  141. package/.agent/scripts/quality-loop-helper.sh +0 -1108
  142. package/.agent/scripts/ralph-loop-helper.sh +0 -836
  143. package/.agent/scripts/ralph-upstream-check.sh +0 -341
  144. package/.agent/scripts/secretlint-helper.sh +0 -847
  145. package/.agent/scripts/servers-helper.sh +0 -241
  146. package/.agent/scripts/ses-helper.sh +0 -619
  147. package/.agent/scripts/session-review-helper.sh +0 -404
  148. package/.agent/scripts/setup-linters-wizard.sh +0 -379
  149. package/.agent/scripts/setup-local-api-keys.sh +0 -330
  150. package/.agent/scripts/setup-mcp-integrations.sh +0 -472
  151. package/.agent/scripts/shared-constants.sh +0 -246
  152. package/.agent/scripts/site-crawler-helper.sh +0 -1487
  153. package/.agent/scripts/snyk-helper.sh +0 -940
  154. package/.agent/scripts/sonarcloud-autofix.sh +0 -193
  155. package/.agent/scripts/sonarcloud-cli.sh +0 -191
  156. package/.agent/scripts/sonarscanner-cli.sh +0 -455
  157. package/.agent/scripts/spaceship-helper.sh +0 -747
  158. package/.agent/scripts/stagehand-helper.sh +0 -321
  159. package/.agent/scripts/stagehand-python-helper.sh +0 -321
  160. package/.agent/scripts/stagehand-python-setup.sh +0 -441
  161. package/.agent/scripts/stagehand-setup.sh +0 -439
  162. package/.agent/scripts/system-cleanup.sh +0 -340
  163. package/.agent/scripts/terminal-title-helper.sh +0 -388
  164. package/.agent/scripts/terminal-title-setup.sh +0 -549
  165. package/.agent/scripts/test-stagehand-both-integration.sh +0 -317
  166. package/.agent/scripts/test-stagehand-integration.sh +0 -309
  167. package/.agent/scripts/test-stagehand-python-integration.sh +0 -341
  168. package/.agent/scripts/todo-ready.sh +0 -263
  169. package/.agent/scripts/tool-version-check.sh +0 -362
  170. package/.agent/scripts/toon-helper.sh +0 -469
  171. package/.agent/scripts/twilio-helper.sh +0 -917
  172. package/.agent/scripts/updown-helper.sh +0 -279
  173. package/.agent/scripts/validate-mcp-integrations.sh +0 -250
  174. package/.agent/scripts/validate-version-consistency.sh +0 -131
  175. package/.agent/scripts/vaultwarden-helper.sh +0 -597
  176. package/.agent/scripts/vercel-cli-helper.sh +0 -816
  177. package/.agent/scripts/verify-mirrors.sh +0 -169
  178. package/.agent/scripts/version-manager.sh +0 -831
  179. package/.agent/scripts/webhosting-helper.sh +0 -471
  180. package/.agent/scripts/webhosting-verify.sh +0 -238
  181. package/.agent/scripts/wordpress-mcp-helper.sh +0 -508
  182. package/.agent/scripts/worktree-helper.sh +0 -595
  183. package/.agent/scripts/worktree-sessions.sh +0 -577
  184. package/.agent/seo/dataforseo.md +0 -215
  185. package/.agent/seo/domain-research.md +0 -532
  186. package/.agent/seo/eeat-score.md +0 -659
  187. package/.agent/seo/google-search-console.md +0 -366
  188. package/.agent/seo/gsc-sitemaps.md +0 -282
  189. package/.agent/seo/keyword-research.md +0 -521
  190. package/.agent/seo/serper.md +0 -278
  191. package/.agent/seo/site-crawler.md +0 -387
  192. package/.agent/seo.md +0 -236
  193. package/.agent/services/accounting/quickfile.md +0 -159
  194. package/.agent/services/communications/telfon.md +0 -470
  195. package/.agent/services/communications/twilio.md +0 -569
  196. package/.agent/services/crm/fluentcrm.md +0 -449
  197. package/.agent/services/email/ses.md +0 -399
  198. package/.agent/services/hosting/101domains.md +0 -378
  199. package/.agent/services/hosting/closte.md +0 -177
  200. package/.agent/services/hosting/cloudflare.md +0 -251
  201. package/.agent/services/hosting/cloudron.md +0 -478
  202. package/.agent/services/hosting/dns-providers.md +0 -335
  203. package/.agent/services/hosting/domain-purchasing.md +0 -344
  204. package/.agent/services/hosting/hetzner.md +0 -327
  205. package/.agent/services/hosting/hostinger.md +0 -287
  206. package/.agent/services/hosting/localhost.md +0 -419
  207. package/.agent/services/hosting/spaceship.md +0 -353
  208. package/.agent/services/hosting/webhosting.md +0 -330
  209. package/.agent/social-media.md +0 -69
  210. package/.agent/templates/plans-template.md +0 -114
  211. package/.agent/templates/prd-template.md +0 -129
  212. package/.agent/templates/tasks-template.md +0 -108
  213. package/.agent/templates/todo-template.md +0 -89
  214. package/.agent/tools/ai-assistants/agno.md +0 -471
  215. package/.agent/tools/ai-assistants/capsolver.md +0 -326
  216. package/.agent/tools/ai-assistants/configuration.md +0 -221
  217. package/.agent/tools/ai-assistants/overview.md +0 -209
  218. package/.agent/tools/ai-assistants/status.md +0 -171
  219. package/.agent/tools/ai-assistants/windsurf.md +0 -193
  220. package/.agent/tools/ai-orchestration/autogen.md +0 -406
  221. package/.agent/tools/ai-orchestration/crewai.md +0 -445
  222. package/.agent/tools/ai-orchestration/langflow.md +0 -405
  223. package/.agent/tools/ai-orchestration/openprose.md +0 -487
  224. package/.agent/tools/ai-orchestration/overview.md +0 -362
  225. package/.agent/tools/ai-orchestration/packaging.md +0 -647
  226. package/.agent/tools/browser/agent-browser.md +0 -464
  227. package/.agent/tools/browser/browser-automation.md +0 -400
  228. package/.agent/tools/browser/chrome-devtools.md +0 -282
  229. package/.agent/tools/browser/crawl4ai-integration.md +0 -422
  230. package/.agent/tools/browser/crawl4ai-resources.md +0 -277
  231. package/.agent/tools/browser/crawl4ai-usage.md +0 -416
  232. package/.agent/tools/browser/crawl4ai.md +0 -585
  233. package/.agent/tools/browser/dev-browser.md +0 -341
  234. package/.agent/tools/browser/pagespeed.md +0 -260
  235. package/.agent/tools/browser/playwright.md +0 -266
  236. package/.agent/tools/browser/playwriter.md +0 -310
  237. package/.agent/tools/browser/stagehand-examples.md +0 -456
  238. package/.agent/tools/browser/stagehand-python.md +0 -483
  239. package/.agent/tools/browser/stagehand.md +0 -421
  240. package/.agent/tools/build-agent/agent-review.md +0 -224
  241. package/.agent/tools/build-agent/build-agent.md +0 -784
  242. package/.agent/tools/build-mcp/aidevops-plugin.md +0 -476
  243. package/.agent/tools/build-mcp/api-wrapper.md +0 -445
  244. package/.agent/tools/build-mcp/build-mcp.md +0 -240
  245. package/.agent/tools/build-mcp/deployment.md +0 -401
  246. package/.agent/tools/build-mcp/server-patterns.md +0 -632
  247. package/.agent/tools/build-mcp/transports.md +0 -366
  248. package/.agent/tools/code-review/auditing.md +0 -383
  249. package/.agent/tools/code-review/automation.md +0 -219
  250. package/.agent/tools/code-review/best-practices.md +0 -203
  251. package/.agent/tools/code-review/codacy.md +0 -151
  252. package/.agent/tools/code-review/code-simplifier.md +0 -174
  253. package/.agent/tools/code-review/code-standards.md +0 -309
  254. package/.agent/tools/code-review/coderabbit.md +0 -101
  255. package/.agent/tools/code-review/management.md +0 -155
  256. package/.agent/tools/code-review/qlty.md +0 -248
  257. package/.agent/tools/code-review/secretlint.md +0 -565
  258. package/.agent/tools/code-review/setup.md +0 -250
  259. package/.agent/tools/code-review/snyk.md +0 -563
  260. package/.agent/tools/code-review/tools.md +0 -230
  261. package/.agent/tools/content/summarize.md +0 -353
  262. package/.agent/tools/context/augment-context-engine.md +0 -468
  263. package/.agent/tools/context/context-builder-agent.md +0 -76
  264. package/.agent/tools/context/context-builder.md +0 -375
  265. package/.agent/tools/context/context7.md +0 -371
  266. package/.agent/tools/context/dspy.md +0 -302
  267. package/.agent/tools/context/dspyground.md +0 -374
  268. package/.agent/tools/context/llm-tldr.md +0 -219
  269. package/.agent/tools/context/osgrep.md +0 -488
  270. package/.agent/tools/context/prompt-optimization.md +0 -338
  271. package/.agent/tools/context/toon.md +0 -292
  272. package/.agent/tools/conversion/pandoc.md +0 -304
  273. package/.agent/tools/credentials/api-key-management.md +0 -154
  274. package/.agent/tools/credentials/api-key-setup.md +0 -224
  275. package/.agent/tools/credentials/environment-variables.md +0 -180
  276. package/.agent/tools/credentials/vaultwarden.md +0 -382
  277. package/.agent/tools/data-extraction/outscraper.md +0 -974
  278. package/.agent/tools/deployment/coolify-cli.md +0 -388
  279. package/.agent/tools/deployment/coolify-setup.md +0 -353
  280. package/.agent/tools/deployment/coolify.md +0 -345
  281. package/.agent/tools/deployment/vercel.md +0 -390
  282. package/.agent/tools/git/authentication.md +0 -132
  283. package/.agent/tools/git/gitea-cli.md +0 -193
  284. package/.agent/tools/git/github-actions.md +0 -207
  285. package/.agent/tools/git/github-cli.md +0 -223
  286. package/.agent/tools/git/gitlab-cli.md +0 -190
  287. package/.agent/tools/git/opencode-github-security.md +0 -350
  288. package/.agent/tools/git/opencode-github.md +0 -328
  289. package/.agent/tools/git/opencode-gitlab.md +0 -252
  290. package/.agent/tools/git/security.md +0 -196
  291. package/.agent/tools/git.md +0 -207
  292. package/.agent/tools/opencode/oh-my-opencode.md +0 -375
  293. package/.agent/tools/opencode/opencode-anthropic-auth.md +0 -446
  294. package/.agent/tools/opencode/opencode.md +0 -651
  295. package/.agent/tools/social-media/bird.md +0 -437
  296. package/.agent/tools/task-management/beads.md +0 -336
  297. package/.agent/tools/terminal/terminal-title.md +0 -251
  298. package/.agent/tools/ui/shadcn.md +0 -196
  299. package/.agent/tools/ui/ui-skills.md +0 -115
  300. package/.agent/tools/wordpress/localwp.md +0 -311
  301. package/.agent/tools/wordpress/mainwp.md +0 -391
  302. package/.agent/tools/wordpress/scf.md +0 -527
  303. package/.agent/tools/wordpress/wp-admin.md +0 -729
  304. package/.agent/tools/wordpress/wp-dev.md +0 -940
  305. package/.agent/tools/wordpress/wp-preferred.md +0 -398
  306. package/.agent/tools/wordpress.md +0 -95
  307. package/.agent/workflows/branch/bugfix.md +0 -63
  308. package/.agent/workflows/branch/chore.md +0 -95
  309. package/.agent/workflows/branch/experiment.md +0 -115
  310. package/.agent/workflows/branch/feature.md +0 -59
  311. package/.agent/workflows/branch/hotfix.md +0 -98
  312. package/.agent/workflows/branch/refactor.md +0 -92
  313. package/.agent/workflows/branch/release.md +0 -96
  314. package/.agent/workflows/branch.md +0 -347
  315. package/.agent/workflows/bug-fixing.md +0 -267
  316. package/.agent/workflows/changelog.md +0 -129
  317. package/.agent/workflows/code-audit-remote.md +0 -279
  318. package/.agent/workflows/conversation-starter.md +0 -69
  319. package/.agent/workflows/error-feedback.md +0 -578
  320. package/.agent/workflows/feature-development.md +0 -355
  321. package/.agent/workflows/git-workflow.md +0 -702
  322. package/.agent/workflows/multi-repo-workspace.md +0 -268
  323. package/.agent/workflows/plans.md +0 -709
  324. package/.agent/workflows/postflight.md +0 -604
  325. package/.agent/workflows/pr.md +0 -571
  326. package/.agent/workflows/preflight.md +0 -278
  327. package/.agent/workflows/ralph-loop.md +0 -773
  328. package/.agent/workflows/release.md +0 -498
  329. package/.agent/workflows/session-manager.md +0 -254
  330. package/.agent/workflows/session-review.md +0 -311
  331. package/.agent/workflows/sql-migrations.md +0 -631
  332. package/.agent/workflows/version-bump.md +0 -283
  333. package/.agent/workflows/wiki-update.md +0 -333
  334. package/.agent/workflows/worktree.md +0 -477
@@ -1,678 +0,0 @@
1
- #!/usr/bin/env bash
2
- # Google Search Console - Sitemap Submission Helper
3
- # Uses Playwright to automate sitemap submissions to GSC
4
- # Part of AI DevOps Framework
5
-
6
- set -euo pipefail
7
-
8
- # Source shared constants
9
- SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
10
- # shellcheck source=shared-constants.sh
11
- source "${SCRIPT_DIR}/shared-constants.sh" 2>/dev/null || true
12
-
13
- # Configuration
14
- readonly CONFIG_FILE="${HOME}/.config/aidevops/gsc-config.json"
15
- readonly WORK_DIR="${HOME}/.aidevops/.agent-workspace/tmp"
16
- readonly GSC_SCRIPT="${WORK_DIR}/gsc-sitemap-submit.js"
17
- readonly CHROME_PROFILE="${HOME}/.aidevops/.agent-workspace/chrome-gsc-profile"
18
- readonly SCREENSHOT_DIR="/tmp/gsc-screenshots"
19
- readonly DEFAULT_SITEMAP="sitemap.xml"
20
-
21
- # Colors
22
- readonly RED='\033[0;31m'
23
- readonly GREEN='\033[0;32m'
24
- readonly YELLOW='\033[1;33m'
25
- readonly BLUE='\033[0;34m'
26
- readonly NC='\033[0m'
27
-
28
- log_info() { echo -e "${BLUE}[INFO]${NC} $1"; }
29
- log_success() { echo -e "${GREEN}[SUCCESS]${NC} $1"; }
30
- log_warn() { echo -e "${YELLOW}[WARN]${NC} $1"; }
31
- log_error() { echo -e "${RED}[ERROR]${NC} $1" >&2; }
32
-
33
- show_help() {
34
- cat << 'HELP'
35
- Usage: gsc-sitemap-helper.sh <command> [options]
36
-
37
- Commands:
38
- submit <domain...> Submit sitemap for one or more domains
39
- submit --file <file> Submit sitemaps for domains listed in file
40
- status <domain> Check sitemap status for a domain
41
- list <domain> List all sitemaps for a domain
42
- login Open browser to login to Google (first-time setup)
43
- setup Install dependencies (Node.js, Playwright)
44
- help Show this help message
45
-
46
- Options:
47
- --sitemap <path> Custom sitemap path (default: sitemap.xml)
48
- --dry-run Show what would be done without making changes
49
- --skip-existing Skip domains that already have sitemaps
50
- --headless Run in headless mode (after initial login)
51
- --timeout <ms> Timeout in milliseconds (default: 60000)
52
-
53
- Examples:
54
- gsc-sitemap-helper.sh submit example.com
55
- gsc-sitemap-helper.sh submit example.com example.net example.org
56
- gsc-sitemap-helper.sh submit --file domains.txt
57
- gsc-sitemap-helper.sh submit example.com --sitemap news-sitemap.xml
58
- gsc-sitemap-helper.sh status example.com
59
- gsc-sitemap-helper.sh login
60
-
61
- Requirements:
62
- - Node.js and npm installed
63
- - Playwright: npm install playwright
64
- - Chrome browser installed
65
- - User logged into Google in the Chrome profile
66
-
67
- First-time setup:
68
- 1. Run: gsc-sitemap-helper.sh setup
69
- 2. Run: gsc-sitemap-helper.sh login
70
- 3. Log into Google in the browser that opens
71
- 4. Close browser when done
72
- 5. Now you can submit sitemaps
73
- HELP
74
- }
75
-
76
- get_chrome_profile_path() {
77
- echo "${CHROME_PROFILE}"
78
- return 0
79
- }
80
-
81
- ensure_directories() {
82
- mkdir -p "${WORK_DIR}"
83
- mkdir -p "${CHROME_PROFILE}"
84
- mkdir -p "${SCREENSHOT_DIR}"
85
- return 0
86
- }
87
-
88
- ensure_playwright() {
89
- if ! command -v npx &> /dev/null; then
90
- log_error "npx not found. Please install Node.js"
91
- return 1
92
- fi
93
-
94
- # Check if playwright is available in WORK_DIR
95
- if [[ ! -d "${WORK_DIR}/node_modules/playwright" ]]; then
96
- log_info "Installing Playwright in ${WORK_DIR}..."
97
- # Install in WORK_DIR to avoid polluting other projects
98
- npm --prefix "${WORK_DIR}" install playwright > /dev/null 2>&1 || {
99
- log_error "Failed to install Playwright"
100
- return 1
101
- }
102
- fi
103
- return 0
104
- }
105
-
106
- # Sanitize domain for safe embedding in JavaScript
107
- sanitize_domain() {
108
- local domain="$1"
109
- # Remove any characters that could break JS string literals
110
- # Allow only alphanumeric, dots, and hyphens (valid domain chars)
111
- echo "$domain" | tr -cd 'a-zA-Z0-9.-'
112
- }
113
-
114
- # Sanitize sitemap path for safe embedding in JavaScript
115
- sanitize_sitemap_path() {
116
- local path="$1"
117
- # Allow only alphanumeric, dots, hyphens, underscores, and forward slashes
118
- # These are valid URL path characters for sitemaps
119
- echo "$path" | tr -cd 'a-zA-Z0-9./_-'
120
- }
121
-
122
- create_submit_script() {
123
- local domains_json="$1"
124
- local sitemap_path
125
- sitemap_path="$(sanitize_sitemap_path "$2")"
126
- local dry_run="$3"
127
- local headless="$4"
128
- local timeout="$5"
129
-
130
- local chrome_profile
131
- chrome_profile="$(get_chrome_profile_path)"
132
-
133
- cat > "${GSC_SCRIPT}" << SCRIPT
134
- import { chromium } from 'playwright';
135
-
136
- const DOMAINS = ${domains_json};
137
- const SITEMAP_PATH = "${sitemap_path}";
138
- const DRY_RUN = ${dry_run};
139
- const HEADLESS = ${headless};
140
- const TIMEOUT = ${timeout};
141
- const SCREENSHOT_DIR = "${SCREENSHOT_DIR}";
142
-
143
- async function waitForGSCLoad(page) {
144
- await page.waitForLoadState('networkidle', { timeout: 30000 }).catch(() => {});
145
- await page.waitForTimeout(2000);
146
- }
147
-
148
- async function main() {
149
- console.log("=".repeat(60));
150
- console.log("Google Search Console - Sitemap Submission Tool");
151
- console.log("=".repeat(60));
152
-
153
- const browser = await chromium.launchPersistentContext(
154
- '${chrome_profile}',
155
- {
156
- headless: HEADLESS,
157
- channel: 'chrome',
158
- ignoreDefaultArgs: ['--enable-automation'],
159
- args: [
160
- '--disable-blink-features=AutomationControlled',
161
- '--disable-infobars',
162
- '--no-first-run',
163
- '--no-default-browser-check'
164
- ],
165
- viewport: { width: 1400, height: 900 }
166
- }
167
- );
168
-
169
- const page = await browser.newPage();
170
- const results = { success: [], skipped: [], failed: [] };
171
-
172
- for (let i = 0; i < DOMAINS.length; i++) {
173
- const domain = DOMAINS[i];
174
- console.log(\`\\n\${'─'.repeat(60)}\`);
175
- console.log(\`[\${i + 1}/\${DOMAINS.length}] \${domain}\`);
176
- console.log('─'.repeat(60));
177
-
178
- try {
179
- const gscUrl = \`https://search.google.com/search-console/sitemaps?resource_id=sc-domain:\${domain}\`;
180
- console.log(\`Opening: \${gscUrl}\`);
181
- await page.goto(gscUrl, { waitUntil: 'domcontentloaded', timeout: TIMEOUT });
182
- await waitForGSCLoad(page);
183
-
184
- const pageContent = await page.content();
185
- if (pageContent.includes("don't have access") || pageContent.includes("Request access")) {
186
- console.log(\`⏭ No access to \${domain}\`);
187
- results.failed.push(\`\${domain} (no access)\`);
188
- continue;
189
- }
190
-
191
- // Check for existing sitemap in table
192
- const sitemapInTable = await page.\$('table:has-text("' + SITEMAP_PATH + '")') ||
193
- await page.\$('tr:has-text("' + SITEMAP_PATH + '")') ||
194
- await page.\$('[role="row"]:has-text("' + SITEMAP_PATH + '")') ||
195
- await page.\$('a:has-text("' + SITEMAP_PATH + '")');
196
-
197
- if (sitemapInTable) {
198
- console.log(\`⏭ Sitemap already submitted for \${domain}\`);
199
- results.skipped.push(domain);
200
- continue;
201
- }
202
-
203
- if (DRY_RUN) {
204
- console.log(\`🔍 Would submit sitemap for \${domain} (dry-run)\`);
205
- results.success.push(\`\${domain} (dry-run)\`);
206
- continue;
207
- }
208
-
209
- // Find the "Add a new sitemap" input
210
- console.log("Looking for 'Add a new sitemap' input...");
211
- const addSitemapSection = await page.\$('text="Add a new sitemap"');
212
-
213
- let input = null;
214
- if (addSitemapSection) {
215
- const container = await addSitemapSection.evaluateHandle(el => {
216
- let parent = el.parentElement;
217
- for (let i = 0; i < 5; i++) {
218
- if (parent && parent.querySelector('input')) {
219
- return parent;
220
- }
221
- parent = parent?.parentElement;
222
- }
223
- return parent;
224
- });
225
- input = await container.\$('input');
226
- }
227
-
228
- // Fallback: find input that's not search
229
- if (!input) {
230
- const allInputs = await page.\$\$('input[type="text"]');
231
- for (const inp of allInputs) {
232
- const placeholder = await inp.getAttribute('placeholder') || '';
233
- const ariaLabel = await inp.getAttribute('aria-label') || '';
234
- if (placeholder.toLowerCase().includes('search') ||
235
- ariaLabel.toLowerCase().includes('search') ||
236
- placeholder.toLowerCase().includes('filter')) {
237
- continue;
238
- }
239
- input = inp;
240
- break;
241
- }
242
- }
243
-
244
- if (!input) {
245
- console.log(\`✗ No 'Add sitemap' input found for \${domain}\`);
246
- await page.screenshot({ path: \`\${SCREENSHOT_DIR}/\${domain.replace(/\\./g, '-')}-error.png\`, fullPage: true });
247
- results.failed.push(\`\${domain} (no input)\`);
248
- continue;
249
- }
250
-
251
- // Fill with full URL - use domain as-is (don't force www.)
252
- // GSC accepts both www and non-www depending on how the property is verified
253
- const fullSitemapUrl = \`https://\${domain}/\${SITEMAP_PATH}\`;
254
- console.log(\`Found input, filling \${fullSitemapUrl}...\`);
255
- await input.click();
256
- await page.waitForTimeout(300);
257
- await input.fill(fullSitemapUrl);
258
- await page.waitForTimeout(500);
259
-
260
- await page.screenshot({ path: \`\${SCREENSHOT_DIR}/\${domain.replace(/\\./g, '-')}-filled.png\`, fullPage: true });
261
-
262
- // Find and click SUBMIT button relative to input
263
- console.log("Clicking SUBMIT button...");
264
- try {
265
- await page.waitForTimeout(500);
266
-
267
- const submitBtn = await input.evaluateHandle(el => {
268
- let parent = el.parentElement;
269
- for (let i = 0; i < 10; i++) {
270
- if (!parent) break;
271
- const btn = parent.querySelector('[role="button"]');
272
- if (btn && btn.textContent.trim().toUpperCase() === 'SUBMIT') {
273
- return btn;
274
- }
275
- parent = parent.parentElement;
276
- }
277
- return null;
278
- });
279
-
280
- if (submitBtn) {
281
- await submitBtn.click();
282
- } else {
283
- throw new Error("Could not find SUBMIT button near input");
284
- }
285
-
286
- await page.waitForTimeout(3000);
287
- await waitForGSCLoad(page);
288
-
289
- await page.screenshot({ path: \`\${SCREENSHOT_DIR}/\${domain.replace(/\\./g, '-')}-submitted.png\`, fullPage: true });
290
- console.log(\`✓ Submitted sitemap for \${domain}\`);
291
- results.success.push(domain);
292
-
293
- } catch (e) {
294
- console.log(\`Submit failed: \${e.message}\`);
295
- await page.screenshot({ path: \`\${SCREENSHOT_DIR}/\${domain.replace(/\\./g, '-')}-submit-error.png\`, fullPage: true });
296
- results.failed.push(\`\${domain} (submit failed)\`);
297
- }
298
-
299
- } catch (error) {
300
- console.error(\`✗ Error for \${domain}: \${error.message}\`);
301
- results.failed.push(\`\${domain} (error)\`);
302
- }
303
-
304
- await page.waitForTimeout(1000);
305
- }
306
-
307
- // Summary
308
- console.log("\\n" + "=".repeat(60));
309
- console.log("SUMMARY");
310
- console.log("=".repeat(60));
311
- console.log(\`\\n✓ Submitted: \${results.success.length}\`);
312
- results.success.forEach(d => console.log(\` \${d}\`));
313
- console.log(\`\\n⏭ Already done: \${results.skipped.length}\`);
314
- results.skipped.forEach(d => console.log(\` \${d}\`));
315
- console.log(\`\\n✗ Failed: \${results.failed.length}\`);
316
- results.failed.forEach(d => console.log(\` \${d}\`));
317
-
318
- await browser.close();
319
-
320
- // Exit with error if any failed
321
- if (results.failed.length > 0) {
322
- process.exit(1);
323
- }
324
- }
325
-
326
- main().catch(err => {
327
- console.error(err);
328
- process.exit(1);
329
- });
330
- SCRIPT
331
- return 0
332
- }
333
-
334
- create_status_script() {
335
- local domain
336
- domain="$(sanitize_domain "$1")"
337
- local chrome_profile
338
- chrome_profile="$(get_chrome_profile_path)"
339
-
340
- cat > "${GSC_SCRIPT}" << SCRIPT
341
- import { chromium } from 'playwright';
342
-
343
- const DOMAIN = "${domain}";
344
-
345
- async function main() {
346
- const browser = await chromium.launchPersistentContext(
347
- '${chrome_profile}',
348
- {
349
- headless: false,
350
- channel: 'chrome',
351
- ignoreDefaultArgs: ['--enable-automation'],
352
- args: [
353
- '--disable-blink-features=AutomationControlled',
354
- '--disable-infobars',
355
- '--no-first-run',
356
- '--no-default-browser-check'
357
- ]
358
- }
359
- );
360
-
361
- const page = await browser.newPage();
362
- const gscUrl = \`https://search.google.com/search-console/sitemaps?resource_id=sc-domain:\${DOMAIN}\`;
363
-
364
- console.log(\`Checking sitemap status for \${DOMAIN}...\`);
365
- await page.goto(gscUrl, { waitUntil: 'networkidle', timeout: 60000 });
366
- await page.waitForTimeout(2000);
367
-
368
- const pageContent = await page.content();
369
-
370
- if (pageContent.includes("don't have access")) {
371
- console.log("❌ No access to this property");
372
- await browser.close();
373
- process.exit(1);
374
- }
375
-
376
- // Look for sitemaps in the table
377
- const sitemapRows = await page.\$\$('tr:has-text("sitemap")');
378
-
379
- if (sitemapRows.length === 0) {
380
- console.log("📭 No sitemaps submitted yet");
381
- } else {
382
- console.log(\`📋 Found \${sitemapRows.length} sitemap(s):\`);
383
- for (const row of sitemapRows) {
384
- const text = await row.textContent();
385
- console.log(\` • \${text.trim()}\`);
386
- }
387
- }
388
-
389
- await browser.close();
390
- }
391
-
392
- main().catch(err => {
393
- console.error(err);
394
- process.exit(1);
395
- });
396
- SCRIPT
397
- return 0
398
- }
399
-
400
- create_login_script() {
401
- local chrome_profile
402
- chrome_profile="$(get_chrome_profile_path)"
403
-
404
- cat > "${GSC_SCRIPT}" << SCRIPT
405
- import { chromium } from 'playwright';
406
-
407
- async function main() {
408
- console.log("Opening Chrome for Google login...");
409
- console.log("Please log into your Google account in the browser.");
410
- console.log("Close the browser when done.");
411
-
412
- const browser = await chromium.launchPersistentContext(
413
- '${chrome_profile}',
414
- {
415
- headless: false,
416
- channel: 'chrome',
417
- ignoreDefaultArgs: ['--enable-automation'],
418
- args: [
419
- '--disable-blink-features=AutomationControlled',
420
- '--disable-infobars',
421
- '--no-first-run',
422
- '--no-default-browser-check'
423
- ]
424
- }
425
- );
426
-
427
- const page = await browser.newPage();
428
- await page.goto('https://search.google.com/search-console', { waitUntil: 'networkidle' });
429
-
430
- console.log("\\nBrowser opened. Please:");
431
- console.log("1. Log into your Google account");
432
- console.log("2. Verify you can see your GSC properties");
433
- console.log("3. Close the browser when done");
434
-
435
- // Wait for browser to close
436
- await new Promise(() => {});
437
- }
438
-
439
- main().catch(console.error);
440
- SCRIPT
441
- return 0
442
- }
443
-
444
- run_script() {
445
- cd "${WORK_DIR}" || return 1
446
- node "${GSC_SCRIPT}"
447
- return $?
448
- }
449
-
450
- cmd_submit() {
451
- local domains=()
452
- local sitemap_path="${DEFAULT_SITEMAP}"
453
- local dry_run="false"
454
- local headless="false"
455
- local timeout="60000"
456
- local file=""
457
-
458
- while [[ $# -gt 0 ]]; do
459
- case "$1" in
460
- --sitemap)
461
- if [[ -z "${2:-}" || "$2" == -* ]]; then
462
- log_error "--sitemap requires a value"
463
- return 1
464
- fi
465
- sitemap_path="$2"
466
- shift 2
467
- ;;
468
- --dry-run)
469
- dry_run="true"
470
- shift
471
- ;;
472
- --headless)
473
- headless="true"
474
- shift
475
- ;;
476
- --timeout)
477
- if [[ -z "${2:-}" || "$2" == -* ]]; then
478
- log_error "--timeout requires a value"
479
- return 1
480
- fi
481
- if ! [[ "$2" =~ ^[0-9]+$ ]]; then
482
- log_error "--timeout must be a number (milliseconds)"
483
- return 1
484
- fi
485
- timeout="$2"
486
- shift 2
487
- ;;
488
- --file)
489
- if [[ -z "${2:-}" || "$2" == -* ]]; then
490
- log_error "--file requires a value"
491
- return 1
492
- fi
493
- file="$2"
494
- shift 2
495
- ;;
496
- --skip-existing)
497
- # Already handled by script logic
498
- shift
499
- ;;
500
- -*)
501
- log_error "Unknown option: $1"
502
- return 1
503
- ;;
504
- *)
505
- # Sanitize domain input
506
- domains+=("$(sanitize_domain "$1")")
507
- shift
508
- ;;
509
- esac
510
- done
511
-
512
- # Read domains from file if specified
513
- if [[ -n "$file" ]]; then
514
- if [[ ! -f "$file" ]]; then
515
- log_error "File not found: $file"
516
- return 1
517
- fi
518
- while IFS= read -r line; do
519
- # Skip empty lines and comments
520
- [[ -z "$line" || "$line" =~ ^# ]] && continue
521
- # Sanitize domain from file
522
- local sanitized
523
- sanitized="$(sanitize_domain "$line")"
524
- [[ -n "$sanitized" ]] && domains+=("$sanitized")
525
- done < "$file"
526
- fi
527
-
528
- if [[ ${#domains[@]} -eq 0 ]]; then
529
- log_error "No domains specified"
530
- show_help
531
- return 1
532
- fi
533
-
534
- # Convert domains array to JSON safely using jq
535
- local domains_json
536
- if command -v jq &> /dev/null; then
537
- domains_json=$(printf '%s\n' "${domains[@]}" | jq -R -s -c 'split("\n") | map(select(length > 0))')
538
- else
539
- # Fallback: manual construction with basic escaping
540
- domains_json="["
541
- for i in "${!domains[@]}"; do
542
- [[ $i -gt 0 ]] && domains_json+=","
543
- # Escape quotes and backslashes
544
- local escaped_domain="${domains[$i]//\\/\\\\}"
545
- escaped_domain="${escaped_domain//\"/\\\"}"
546
- domains_json+="\"${escaped_domain}\""
547
- done
548
- domains_json+="]"
549
- fi
550
-
551
- ensure_directories
552
- ensure_playwright || return 1
553
-
554
- log_info "Submitting sitemaps for ${#domains[@]} domain(s)..."
555
- [[ "$dry_run" == "true" ]] && log_warn "DRY RUN - no changes will be made"
556
-
557
- create_submit_script "$domains_json" "$sitemap_path" "$dry_run" "$headless" "$timeout"
558
- run_script
559
- return $?
560
- }
561
-
562
- cmd_status() {
563
- local domain="${1:-}"
564
-
565
- if [[ -z "$domain" ]]; then
566
- log_error "Domain required"
567
- return 1
568
- fi
569
-
570
- ensure_directories
571
- ensure_playwright || return 1
572
-
573
- create_status_script "$domain"
574
- run_script
575
- return $?
576
- }
577
-
578
- cmd_list() {
579
- # Same as status for now
580
- cmd_status "$@"
581
- return $?
582
- }
583
-
584
- cmd_login() {
585
- ensure_directories
586
- ensure_playwright || return 1
587
-
588
- log_info "Opening browser for Google login..."
589
- create_login_script
590
- run_script
591
- return $?
592
- }
593
-
594
- cmd_setup() {
595
- log_info "Setting up GSC Sitemap Helper..."
596
-
597
- ensure_directories
598
-
599
- # Check Node.js
600
- if ! command -v node &> /dev/null; then
601
- log_error "Node.js not found. Please install Node.js first."
602
- log_info "Install with: brew install node (macOS) or see https://nodejs.org"
603
- return 1
604
- fi
605
- log_success "Node.js $(node --version) found"
606
-
607
- # Check npm
608
- if ! command -v npm &> /dev/null; then
609
- log_error "npm not found. Please install npm."
610
- return 1
611
- fi
612
- log_success "npm $(npm --version) found"
613
-
614
- # Install Playwright
615
- log_info "Installing Playwright..."
616
- cd "${WORK_DIR}" || return 1
617
-
618
- if [[ ! -f "package.json" ]]; then
619
- npm init -y > /dev/null 2>&1
620
- fi
621
-
622
- # NOSONAR - npm scripts required for Playwright browser automation binaries
623
- npm install playwright > /dev/null 2>&1
624
- log_success "Playwright installed"
625
-
626
- # Create config file if it doesn't exist
627
- if [[ ! -f "$CONFIG_FILE" ]]; then
628
- mkdir -p "$(dirname "$CONFIG_FILE")"
629
- cat > "$CONFIG_FILE" << 'CONFIG'
630
- {
631
- "chrome_profile_dir": "~/.aidevops/.agent-workspace/chrome-gsc-profile",
632
- "default_sitemap_path": "sitemap.xml",
633
- "screenshot_dir": "/tmp/gsc-screenshots",
634
- "timeout_ms": 60000,
635
- "headless": false
636
- }
637
- CONFIG
638
- log_success "Created config file: $CONFIG_FILE"
639
- fi
640
-
641
- log_success "Setup complete!"
642
- log_info "Next steps:"
643
- log_info " 1. Run: gsc-sitemap-helper.sh login"
644
- log_info " 2. Log into Google in the browser that opens"
645
- log_info " 3. Close browser when done"
646
- log_info " 4. Now you can submit sitemaps!"
647
- return 0
648
- }
649
-
650
- # Main
651
- case "${1:-}" in
652
- submit)
653
- shift
654
- cmd_submit "$@"
655
- ;;
656
- status)
657
- shift
658
- cmd_status "${1:-}"
659
- ;;
660
- list)
661
- shift
662
- cmd_list "${1:-}"
663
- ;;
664
- login)
665
- cmd_login
666
- ;;
667
- setup)
668
- cmd_setup
669
- ;;
670
- -h|--help|help|"")
671
- show_help
672
- ;;
673
- *)
674
- log_error "Unknown command: $1"
675
- show_help
676
- exit 1
677
- ;;
678
- esac