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.
- package/README.md +1 -1
- package/VERSION +1 -1
- package/aidevops.sh +15 -9
- package/package.json +4 -4
- package/scripts/npm-postinstall.js +6 -7
- package/setup.sh +1 -1
- package/templates/deploy-templates.sh +144 -0
- package/templates/home/.agent/README.md +33 -0
- package/templates/home/AGENTS.md +96 -0
- package/templates/home/git/.agent/README.md +48 -0
- package/templates/home/git/AGENTS.md +97 -0
- package/templates/standard-functions.sh +179 -0
- package/templates/wordpress-performance-workflow.md +217 -0
- package/.agent/AGENTS.md +0 -614
- package/.agent/accounts.md +0 -65
- package/.agent/aidevops/add-new-mcp-to-aidevops.md +0 -456
- package/.agent/aidevops/api-integrations.md +0 -335
- package/.agent/aidevops/architecture.md +0 -510
- package/.agent/aidevops/configs.md +0 -274
- package/.agent/aidevops/docs.md +0 -244
- package/.agent/aidevops/extension.md +0 -311
- package/.agent/aidevops/mcp-integrations.md +0 -340
- package/.agent/aidevops/mcp-troubleshooting.md +0 -162
- package/.agent/aidevops/memory-patterns.md +0 -172
- package/.agent/aidevops/providers.md +0 -217
- package/.agent/aidevops/recommendations.md +0 -321
- package/.agent/aidevops/requirements.md +0 -301
- package/.agent/aidevops/resources.md +0 -214
- package/.agent/aidevops/security-requirements.md +0 -174
- package/.agent/aidevops/security.md +0 -350
- package/.agent/aidevops/service-links.md +0 -400
- package/.agent/aidevops/services.md +0 -357
- package/.agent/aidevops/setup.md +0 -153
- package/.agent/aidevops/troubleshooting.md +0 -389
- package/.agent/aidevops.md +0 -124
- package/.agent/build-plus.md +0 -244
- package/.agent/content/guidelines.md +0 -109
- package/.agent/content.md +0 -87
- package/.agent/health.md +0 -59
- package/.agent/legal.md +0 -59
- package/.agent/loop-state/full-loop.local.md +0 -16
- package/.agent/loop-state/ralph-loop.local.md +0 -10
- package/.agent/marketing.md +0 -440
- package/.agent/memory/README.md +0 -260
- package/.agent/onboarding.md +0 -796
- package/.agent/plan-plus.md +0 -245
- package/.agent/research.md +0 -100
- package/.agent/sales.md +0 -333
- package/.agent/scripts/101domains-helper.sh +0 -701
- package/.agent/scripts/add-missing-returns.sh +0 -140
- package/.agent/scripts/agent-browser-helper.sh +0 -311
- package/.agent/scripts/agno-setup.sh +0 -712
- package/.agent/scripts/ahrefs-mcp-wrapper.js +0 -168
- package/.agent/scripts/aidevops-update-check.sh +0 -71
- package/.agent/scripts/ampcode-cli.sh +0 -522
- package/.agent/scripts/auto-version-bump.sh +0 -156
- package/.agent/scripts/autogen-helper.sh +0 -512
- package/.agent/scripts/beads-sync-helper.sh +0 -596
- package/.agent/scripts/closte-helper.sh +0 -5
- package/.agent/scripts/cloudron-helper.sh +0 -321
- package/.agent/scripts/codacy-cli-chunked.sh +0 -581
- package/.agent/scripts/codacy-cli.sh +0 -442
- package/.agent/scripts/code-audit-helper.sh +0 -5
- package/.agent/scripts/coderabbit-cli.sh +0 -417
- package/.agent/scripts/coderabbit-pro-analysis.sh +0 -238
- package/.agent/scripts/commands/code-simplifier.md +0 -86
- package/.agent/scripts/commands/full-loop.md +0 -246
- package/.agent/scripts/commands/postflight-loop.md +0 -103
- package/.agent/scripts/commands/recall.md +0 -182
- package/.agent/scripts/commands/remember.md +0 -132
- package/.agent/scripts/commands/save-todo.md +0 -175
- package/.agent/scripts/commands/session-review.md +0 -154
- package/.agent/scripts/comprehensive-quality-fix.sh +0 -106
- package/.agent/scripts/context-builder-helper.sh +0 -522
- package/.agent/scripts/coolify-cli-helper.sh +0 -674
- package/.agent/scripts/coolify-helper.sh +0 -380
- package/.agent/scripts/crawl4ai-examples.sh +0 -401
- package/.agent/scripts/crawl4ai-helper.sh +0 -1078
- package/.agent/scripts/crewai-helper.sh +0 -681
- package/.agent/scripts/dev-browser-helper.sh +0 -513
- package/.agent/scripts/dns-helper.sh +0 -396
- package/.agent/scripts/domain-research-helper.sh +0 -917
- package/.agent/scripts/dspy-helper.sh +0 -285
- package/.agent/scripts/dspyground-helper.sh +0 -291
- package/.agent/scripts/eeat-score-helper.sh +0 -1242
- package/.agent/scripts/efficient-return-fix.sh +0 -92
- package/.agent/scripts/extract-opencode-prompts.sh +0 -128
- package/.agent/scripts/find-missing-returns.sh +0 -113
- package/.agent/scripts/fix-auth-headers.sh +0 -104
- package/.agent/scripts/fix-common-strings.sh +0 -254
- package/.agent/scripts/fix-content-type.sh +0 -100
- package/.agent/scripts/fix-error-messages.sh +0 -130
- package/.agent/scripts/fix-misplaced-returns.sh +0 -74
- package/.agent/scripts/fix-remaining-literals.sh +0 -152
- package/.agent/scripts/fix-return-statements.sh +0 -41
- package/.agent/scripts/fix-s131-default-cases.sh +0 -249
- package/.agent/scripts/fix-sc2155-simple.sh +0 -102
- package/.agent/scripts/fix-shellcheck-critical.sh +0 -187
- package/.agent/scripts/fix-string-literals.sh +0 -273
- package/.agent/scripts/full-loop-helper.sh +0 -773
- package/.agent/scripts/generate-opencode-agents.sh +0 -497
- package/.agent/scripts/generate-opencode-commands.sh +0 -1629
- package/.agent/scripts/generate-skills.sh +0 -366
- package/.agent/scripts/git-platforms-helper.sh +0 -640
- package/.agent/scripts/gitea-cli-helper.sh +0 -743
- package/.agent/scripts/github-cli-helper.sh +0 -702
- package/.agent/scripts/gitlab-cli-helper.sh +0 -682
- package/.agent/scripts/gsc-add-user-helper.sh +0 -325
- package/.agent/scripts/gsc-sitemap-helper.sh +0 -678
- package/.agent/scripts/hetzner-helper.sh +0 -485
- package/.agent/scripts/hostinger-helper.sh +0 -229
- package/.agent/scripts/keyword-research-helper.sh +0 -1815
- package/.agent/scripts/langflow-helper.sh +0 -544
- package/.agent/scripts/linkedin-automation.py +0 -241
- package/.agent/scripts/linter-manager.sh +0 -599
- package/.agent/scripts/linters-local.sh +0 -434
- package/.agent/scripts/list-keys-helper.sh +0 -488
- package/.agent/scripts/local-browser-automation.py +0 -339
- package/.agent/scripts/localhost-helper.sh +0 -744
- package/.agent/scripts/loop-common.sh +0 -806
- package/.agent/scripts/mainwp-helper.sh +0 -728
- package/.agent/scripts/markdown-formatter.sh +0 -338
- package/.agent/scripts/markdown-lint-fix.sh +0 -311
- package/.agent/scripts/mass-fix-returns.sh +0 -58
- package/.agent/scripts/mcp-diagnose.sh +0 -167
- package/.agent/scripts/mcp-inspector-helper.sh +0 -449
- package/.agent/scripts/memory-helper.sh +0 -650
- package/.agent/scripts/monitor-code-review.sh +0 -255
- package/.agent/scripts/onboarding-helper.sh +0 -706
- package/.agent/scripts/opencode-github-setup-helper.sh +0 -797
- package/.agent/scripts/opencode-test-helper.sh +0 -213
- package/.agent/scripts/pagespeed-helper.sh +0 -464
- package/.agent/scripts/pandoc-helper.sh +0 -362
- package/.agent/scripts/postflight-check.sh +0 -555
- package/.agent/scripts/pre-commit-hook.sh +0 -259
- package/.agent/scripts/pre-edit-check.sh +0 -169
- package/.agent/scripts/qlty-cli.sh +0 -356
- package/.agent/scripts/quality-cli-manager.sh +0 -525
- package/.agent/scripts/quality-feedback-helper.sh +0 -462
- package/.agent/scripts/quality-fix.sh +0 -263
- package/.agent/scripts/quality-loop-helper.sh +0 -1108
- package/.agent/scripts/ralph-loop-helper.sh +0 -836
- package/.agent/scripts/ralph-upstream-check.sh +0 -341
- package/.agent/scripts/secretlint-helper.sh +0 -847
- package/.agent/scripts/servers-helper.sh +0 -241
- package/.agent/scripts/ses-helper.sh +0 -619
- package/.agent/scripts/session-review-helper.sh +0 -404
- package/.agent/scripts/setup-linters-wizard.sh +0 -379
- package/.agent/scripts/setup-local-api-keys.sh +0 -330
- package/.agent/scripts/setup-mcp-integrations.sh +0 -472
- package/.agent/scripts/shared-constants.sh +0 -246
- package/.agent/scripts/site-crawler-helper.sh +0 -1487
- package/.agent/scripts/snyk-helper.sh +0 -940
- package/.agent/scripts/sonarcloud-autofix.sh +0 -193
- package/.agent/scripts/sonarcloud-cli.sh +0 -191
- package/.agent/scripts/sonarscanner-cli.sh +0 -455
- package/.agent/scripts/spaceship-helper.sh +0 -747
- package/.agent/scripts/stagehand-helper.sh +0 -321
- package/.agent/scripts/stagehand-python-helper.sh +0 -321
- package/.agent/scripts/stagehand-python-setup.sh +0 -441
- package/.agent/scripts/stagehand-setup.sh +0 -439
- package/.agent/scripts/system-cleanup.sh +0 -340
- package/.agent/scripts/terminal-title-helper.sh +0 -388
- package/.agent/scripts/terminal-title-setup.sh +0 -549
- package/.agent/scripts/test-stagehand-both-integration.sh +0 -317
- package/.agent/scripts/test-stagehand-integration.sh +0 -309
- package/.agent/scripts/test-stagehand-python-integration.sh +0 -341
- package/.agent/scripts/todo-ready.sh +0 -263
- package/.agent/scripts/tool-version-check.sh +0 -362
- package/.agent/scripts/toon-helper.sh +0 -469
- package/.agent/scripts/twilio-helper.sh +0 -917
- package/.agent/scripts/updown-helper.sh +0 -279
- package/.agent/scripts/validate-mcp-integrations.sh +0 -250
- package/.agent/scripts/validate-version-consistency.sh +0 -131
- package/.agent/scripts/vaultwarden-helper.sh +0 -597
- package/.agent/scripts/vercel-cli-helper.sh +0 -816
- package/.agent/scripts/verify-mirrors.sh +0 -169
- package/.agent/scripts/version-manager.sh +0 -831
- package/.agent/scripts/webhosting-helper.sh +0 -471
- package/.agent/scripts/webhosting-verify.sh +0 -238
- package/.agent/scripts/wordpress-mcp-helper.sh +0 -508
- package/.agent/scripts/worktree-helper.sh +0 -595
- package/.agent/scripts/worktree-sessions.sh +0 -577
- package/.agent/seo/dataforseo.md +0 -215
- package/.agent/seo/domain-research.md +0 -532
- package/.agent/seo/eeat-score.md +0 -659
- package/.agent/seo/google-search-console.md +0 -366
- package/.agent/seo/gsc-sitemaps.md +0 -282
- package/.agent/seo/keyword-research.md +0 -521
- package/.agent/seo/serper.md +0 -278
- package/.agent/seo/site-crawler.md +0 -387
- package/.agent/seo.md +0 -236
- package/.agent/services/accounting/quickfile.md +0 -159
- package/.agent/services/communications/telfon.md +0 -470
- package/.agent/services/communications/twilio.md +0 -569
- package/.agent/services/crm/fluentcrm.md +0 -449
- package/.agent/services/email/ses.md +0 -399
- package/.agent/services/hosting/101domains.md +0 -378
- package/.agent/services/hosting/closte.md +0 -177
- package/.agent/services/hosting/cloudflare.md +0 -251
- package/.agent/services/hosting/cloudron.md +0 -478
- package/.agent/services/hosting/dns-providers.md +0 -335
- package/.agent/services/hosting/domain-purchasing.md +0 -344
- package/.agent/services/hosting/hetzner.md +0 -327
- package/.agent/services/hosting/hostinger.md +0 -287
- package/.agent/services/hosting/localhost.md +0 -419
- package/.agent/services/hosting/spaceship.md +0 -353
- package/.agent/services/hosting/webhosting.md +0 -330
- package/.agent/social-media.md +0 -69
- package/.agent/templates/plans-template.md +0 -114
- package/.agent/templates/prd-template.md +0 -129
- package/.agent/templates/tasks-template.md +0 -108
- package/.agent/templates/todo-template.md +0 -89
- package/.agent/tools/ai-assistants/agno.md +0 -471
- package/.agent/tools/ai-assistants/capsolver.md +0 -326
- package/.agent/tools/ai-assistants/configuration.md +0 -221
- package/.agent/tools/ai-assistants/overview.md +0 -209
- package/.agent/tools/ai-assistants/status.md +0 -171
- package/.agent/tools/ai-assistants/windsurf.md +0 -193
- package/.agent/tools/ai-orchestration/autogen.md +0 -406
- package/.agent/tools/ai-orchestration/crewai.md +0 -445
- package/.agent/tools/ai-orchestration/langflow.md +0 -405
- package/.agent/tools/ai-orchestration/openprose.md +0 -487
- package/.agent/tools/ai-orchestration/overview.md +0 -362
- package/.agent/tools/ai-orchestration/packaging.md +0 -647
- package/.agent/tools/browser/agent-browser.md +0 -464
- package/.agent/tools/browser/browser-automation.md +0 -400
- package/.agent/tools/browser/chrome-devtools.md +0 -282
- package/.agent/tools/browser/crawl4ai-integration.md +0 -422
- package/.agent/tools/browser/crawl4ai-resources.md +0 -277
- package/.agent/tools/browser/crawl4ai-usage.md +0 -416
- package/.agent/tools/browser/crawl4ai.md +0 -585
- package/.agent/tools/browser/dev-browser.md +0 -341
- package/.agent/tools/browser/pagespeed.md +0 -260
- package/.agent/tools/browser/playwright.md +0 -266
- package/.agent/tools/browser/playwriter.md +0 -310
- package/.agent/tools/browser/stagehand-examples.md +0 -456
- package/.agent/tools/browser/stagehand-python.md +0 -483
- package/.agent/tools/browser/stagehand.md +0 -421
- package/.agent/tools/build-agent/agent-review.md +0 -224
- package/.agent/tools/build-agent/build-agent.md +0 -784
- package/.agent/tools/build-mcp/aidevops-plugin.md +0 -476
- package/.agent/tools/build-mcp/api-wrapper.md +0 -445
- package/.agent/tools/build-mcp/build-mcp.md +0 -240
- package/.agent/tools/build-mcp/deployment.md +0 -401
- package/.agent/tools/build-mcp/server-patterns.md +0 -632
- package/.agent/tools/build-mcp/transports.md +0 -366
- package/.agent/tools/code-review/auditing.md +0 -383
- package/.agent/tools/code-review/automation.md +0 -219
- package/.agent/tools/code-review/best-practices.md +0 -203
- package/.agent/tools/code-review/codacy.md +0 -151
- package/.agent/tools/code-review/code-simplifier.md +0 -174
- package/.agent/tools/code-review/code-standards.md +0 -309
- package/.agent/tools/code-review/coderabbit.md +0 -101
- package/.agent/tools/code-review/management.md +0 -155
- package/.agent/tools/code-review/qlty.md +0 -248
- package/.agent/tools/code-review/secretlint.md +0 -565
- package/.agent/tools/code-review/setup.md +0 -250
- package/.agent/tools/code-review/snyk.md +0 -563
- package/.agent/tools/code-review/tools.md +0 -230
- package/.agent/tools/content/summarize.md +0 -353
- package/.agent/tools/context/augment-context-engine.md +0 -468
- package/.agent/tools/context/context-builder-agent.md +0 -76
- package/.agent/tools/context/context-builder.md +0 -375
- package/.agent/tools/context/context7.md +0 -371
- package/.agent/tools/context/dspy.md +0 -302
- package/.agent/tools/context/dspyground.md +0 -374
- package/.agent/tools/context/llm-tldr.md +0 -219
- package/.agent/tools/context/osgrep.md +0 -488
- package/.agent/tools/context/prompt-optimization.md +0 -338
- package/.agent/tools/context/toon.md +0 -292
- package/.agent/tools/conversion/pandoc.md +0 -304
- package/.agent/tools/credentials/api-key-management.md +0 -154
- package/.agent/tools/credentials/api-key-setup.md +0 -224
- package/.agent/tools/credentials/environment-variables.md +0 -180
- package/.agent/tools/credentials/vaultwarden.md +0 -382
- package/.agent/tools/data-extraction/outscraper.md +0 -974
- package/.agent/tools/deployment/coolify-cli.md +0 -388
- package/.agent/tools/deployment/coolify-setup.md +0 -353
- package/.agent/tools/deployment/coolify.md +0 -345
- package/.agent/tools/deployment/vercel.md +0 -390
- package/.agent/tools/git/authentication.md +0 -132
- package/.agent/tools/git/gitea-cli.md +0 -193
- package/.agent/tools/git/github-actions.md +0 -207
- package/.agent/tools/git/github-cli.md +0 -223
- package/.agent/tools/git/gitlab-cli.md +0 -190
- package/.agent/tools/git/opencode-github-security.md +0 -350
- package/.agent/tools/git/opencode-github.md +0 -328
- package/.agent/tools/git/opencode-gitlab.md +0 -252
- package/.agent/tools/git/security.md +0 -196
- package/.agent/tools/git.md +0 -207
- package/.agent/tools/opencode/oh-my-opencode.md +0 -375
- package/.agent/tools/opencode/opencode-anthropic-auth.md +0 -446
- package/.agent/tools/opencode/opencode.md +0 -651
- package/.agent/tools/social-media/bird.md +0 -437
- package/.agent/tools/task-management/beads.md +0 -336
- package/.agent/tools/terminal/terminal-title.md +0 -251
- package/.agent/tools/ui/shadcn.md +0 -196
- package/.agent/tools/ui/ui-skills.md +0 -115
- package/.agent/tools/wordpress/localwp.md +0 -311
- package/.agent/tools/wordpress/mainwp.md +0 -391
- package/.agent/tools/wordpress/scf.md +0 -527
- package/.agent/tools/wordpress/wp-admin.md +0 -729
- package/.agent/tools/wordpress/wp-dev.md +0 -940
- package/.agent/tools/wordpress/wp-preferred.md +0 -398
- package/.agent/tools/wordpress.md +0 -95
- package/.agent/workflows/branch/bugfix.md +0 -63
- package/.agent/workflows/branch/chore.md +0 -95
- package/.agent/workflows/branch/experiment.md +0 -115
- package/.agent/workflows/branch/feature.md +0 -59
- package/.agent/workflows/branch/hotfix.md +0 -98
- package/.agent/workflows/branch/refactor.md +0 -92
- package/.agent/workflows/branch/release.md +0 -96
- package/.agent/workflows/branch.md +0 -347
- package/.agent/workflows/bug-fixing.md +0 -267
- package/.agent/workflows/changelog.md +0 -129
- package/.agent/workflows/code-audit-remote.md +0 -279
- package/.agent/workflows/conversation-starter.md +0 -69
- package/.agent/workflows/error-feedback.md +0 -578
- package/.agent/workflows/feature-development.md +0 -355
- package/.agent/workflows/git-workflow.md +0 -702
- package/.agent/workflows/multi-repo-workspace.md +0 -268
- package/.agent/workflows/plans.md +0 -709
- package/.agent/workflows/postflight.md +0 -604
- package/.agent/workflows/pr.md +0 -571
- package/.agent/workflows/preflight.md +0 -278
- package/.agent/workflows/ralph-loop.md +0 -773
- package/.agent/workflows/release.md +0 -498
- package/.agent/workflows/session-manager.md +0 -254
- package/.agent/workflows/session-review.md +0 -311
- package/.agent/workflows/sql-migrations.md +0 -631
- package/.agent/workflows/version-bump.md +0 -283
- package/.agent/workflows/wiki-update.md +0 -333
- package/.agent/workflows/worktree.md +0 -477
|
@@ -1,632 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
description: MCP server patterns for tools, resources, and prompts
|
|
3
|
-
mode: subagent
|
|
4
|
-
tools:
|
|
5
|
-
read: true
|
|
6
|
-
write: true
|
|
7
|
-
edit: true
|
|
8
|
-
bash: true
|
|
9
|
-
glob: true
|
|
10
|
-
grep: true
|
|
11
|
-
webfetch: true
|
|
12
|
-
---
|
|
13
|
-
|
|
14
|
-
# MCP Server Patterns - Tools, Resources, Prompts
|
|
15
|
-
|
|
16
|
-
<!-- AI-CONTEXT-START -->
|
|
17
|
-
|
|
18
|
-
## Quick Reference
|
|
19
|
-
|
|
20
|
-
- **Purpose**: Patterns for registering MCP tools, resources, and prompts
|
|
21
|
-
- **Validation**: Always use Zod schemas with `.describe()`
|
|
22
|
-
- **SDK**: `@modelcontextprotocol/sdk` + `elysia-mcp`
|
|
23
|
-
|
|
24
|
-
**Pattern Types**:
|
|
25
|
-
|
|
26
|
-
| Type | Purpose | Naming Pattern | Example |
|
|
27
|
-
|------|---------|----------------|---------|
|
|
28
|
-
| Tool | Execute actions, call APIs | `verb_noun` | `get_user`, `create_item` |
|
|
29
|
-
| Resource | Expose data/files | `protocol://path` | `config://settings` |
|
|
30
|
-
| Prompt | Reusable prompt templates | `action` or `action_context` | `summarize`, `code_review` |
|
|
31
|
-
|
|
32
|
-
**Tool Naming Quick Reference**:
|
|
33
|
-
|
|
34
|
-
| Verb | Use Case | Examples |
|
|
35
|
-
|------|----------|----------|
|
|
36
|
-
| `get` | Single item by ID | `get_user`, `get_config` |
|
|
37
|
-
| `list` | Multiple items | `list_users`, `list_files` |
|
|
38
|
-
| `search` | Query with filters | `search_docs`, `search_logs` |
|
|
39
|
-
| `create` | New resource | `create_user`, `create_project` |
|
|
40
|
-
| `update` | Modify existing | `update_user`, `update_settings` |
|
|
41
|
-
| `delete` | Remove resource | `delete_user`, `delete_file` |
|
|
42
|
-
|
|
43
|
-
<!-- AI-CONTEXT-END -->
|
|
44
|
-
|
|
45
|
-
## Tool Patterns
|
|
46
|
-
|
|
47
|
-
### Basic Tool
|
|
48
|
-
|
|
49
|
-
```typescript
|
|
50
|
-
import { z } from 'zod';
|
|
51
|
-
|
|
52
|
-
server.tool(
|
|
53
|
-
'greet',
|
|
54
|
-
{
|
|
55
|
-
name: z.string().describe('Name to greet'),
|
|
56
|
-
},
|
|
57
|
-
async (args) => ({
|
|
58
|
-
content: [{ type: 'text', text: `Hello, ${args.name}!` }],
|
|
59
|
-
})
|
|
60
|
-
);
|
|
61
|
-
```
|
|
62
|
-
|
|
63
|
-
### Tool with Optional Parameters
|
|
64
|
-
|
|
65
|
-
```typescript
|
|
66
|
-
server.tool(
|
|
67
|
-
'search',
|
|
68
|
-
{
|
|
69
|
-
query: z.string().describe('Search query'),
|
|
70
|
-
limit: z.number().optional().default(10).describe('Max results'),
|
|
71
|
-
offset: z.number().optional().default(0).describe('Result offset'),
|
|
72
|
-
},
|
|
73
|
-
async (args) => {
|
|
74
|
-
const results = await performSearch(args.query, args.limit, args.offset);
|
|
75
|
-
return {
|
|
76
|
-
content: [{ type: 'text', text: JSON.stringify(results, null, 2) }],
|
|
77
|
-
};
|
|
78
|
-
}
|
|
79
|
-
);
|
|
80
|
-
```
|
|
81
|
-
|
|
82
|
-
### Tool with Enum Validation
|
|
83
|
-
|
|
84
|
-
```typescript
|
|
85
|
-
server.tool(
|
|
86
|
-
'set_status',
|
|
87
|
-
{
|
|
88
|
-
status: z.enum(['active', 'inactive', 'pending']).describe('New status'),
|
|
89
|
-
reason: z.string().optional().describe('Reason for change'),
|
|
90
|
-
},
|
|
91
|
-
async (args) => {
|
|
92
|
-
await updateStatus(args.status, args.reason);
|
|
93
|
-
return {
|
|
94
|
-
content: [{ type: 'text', text: `Status updated to: ${args.status}` }],
|
|
95
|
-
};
|
|
96
|
-
}
|
|
97
|
-
);
|
|
98
|
-
```
|
|
99
|
-
|
|
100
|
-
### Tool with Complex Input
|
|
101
|
-
|
|
102
|
-
```typescript
|
|
103
|
-
const AddressSchema = z.object({
|
|
104
|
-
street: z.string(),
|
|
105
|
-
city: z.string(),
|
|
106
|
-
country: z.string(),
|
|
107
|
-
postal: z.string().optional(),
|
|
108
|
-
});
|
|
109
|
-
|
|
110
|
-
server.tool(
|
|
111
|
-
'create_contact',
|
|
112
|
-
{
|
|
113
|
-
name: z.string().describe('Contact name'),
|
|
114
|
-
email: z.string().email().describe('Email address'),
|
|
115
|
-
phone: z.string().optional().describe('Phone number'),
|
|
116
|
-
address: AddressSchema.optional().describe('Mailing address'),
|
|
117
|
-
},
|
|
118
|
-
async (args) => {
|
|
119
|
-
const contact = await createContact(args);
|
|
120
|
-
return {
|
|
121
|
-
content: [{ type: 'text', text: JSON.stringify(contact) }],
|
|
122
|
-
};
|
|
123
|
-
}
|
|
124
|
-
);
|
|
125
|
-
```
|
|
126
|
-
|
|
127
|
-
### Tool with Structured Output
|
|
128
|
-
|
|
129
|
-
```typescript
|
|
130
|
-
server.registerTool(
|
|
131
|
-
'calculate',
|
|
132
|
-
{
|
|
133
|
-
title: 'Calculator',
|
|
134
|
-
description: 'Perform mathematical calculations',
|
|
135
|
-
inputSchema: {
|
|
136
|
-
expression: z.string().describe('Math expression to evaluate'),
|
|
137
|
-
},
|
|
138
|
-
outputSchema: {
|
|
139
|
-
result: z.number(),
|
|
140
|
-
expression: z.string(),
|
|
141
|
-
timestamp: z.string(),
|
|
142
|
-
},
|
|
143
|
-
},
|
|
144
|
-
async ({ expression }) => {
|
|
145
|
-
const result = evaluate(expression);
|
|
146
|
-
const output = {
|
|
147
|
-
result,
|
|
148
|
-
expression,
|
|
149
|
-
timestamp: new Date().toISOString(),
|
|
150
|
-
};
|
|
151
|
-
return {
|
|
152
|
-
content: [{ type: 'text', text: JSON.stringify(output) }],
|
|
153
|
-
structuredContent: output,
|
|
154
|
-
};
|
|
155
|
-
}
|
|
156
|
-
);
|
|
157
|
-
```
|
|
158
|
-
|
|
159
|
-
### Tool with Error Handling
|
|
160
|
-
|
|
161
|
-
```typescript
|
|
162
|
-
server.tool(
|
|
163
|
-
'fetch_data',
|
|
164
|
-
{
|
|
165
|
-
url: z.string().url().describe('URL to fetch'),
|
|
166
|
-
},
|
|
167
|
-
async (args) => {
|
|
168
|
-
try {
|
|
169
|
-
const response = await fetch(args.url);
|
|
170
|
-
if (!response.ok) {
|
|
171
|
-
return {
|
|
172
|
-
content: [{
|
|
173
|
-
type: 'text',
|
|
174
|
-
text: JSON.stringify({
|
|
175
|
-
error: true,
|
|
176
|
-
status: response.status,
|
|
177
|
-
message: `HTTP ${response.status}: ${response.statusText}`,
|
|
178
|
-
}),
|
|
179
|
-
}],
|
|
180
|
-
isError: true,
|
|
181
|
-
};
|
|
182
|
-
}
|
|
183
|
-
const data = await response.json();
|
|
184
|
-
return {
|
|
185
|
-
content: [{ type: 'text', text: JSON.stringify(data) }],
|
|
186
|
-
};
|
|
187
|
-
} catch (error) {
|
|
188
|
-
return {
|
|
189
|
-
content: [{
|
|
190
|
-
type: 'text',
|
|
191
|
-
text: JSON.stringify({
|
|
192
|
-
error: true,
|
|
193
|
-
message: error instanceof Error ? error.message : 'Unknown error',
|
|
194
|
-
}),
|
|
195
|
-
}],
|
|
196
|
-
isError: true,
|
|
197
|
-
};
|
|
198
|
-
}
|
|
199
|
-
}
|
|
200
|
-
);
|
|
201
|
-
```
|
|
202
|
-
|
|
203
|
-
## Resource Patterns
|
|
204
|
-
|
|
205
|
-
### Static Resource
|
|
206
|
-
|
|
207
|
-
```typescript
|
|
208
|
-
server.resource('README', 'resource://readme', async () => ({
|
|
209
|
-
contents: [{
|
|
210
|
-
uri: 'resource://readme',
|
|
211
|
-
mimeType: 'text/markdown',
|
|
212
|
-
text: '# My MCP Server\n\nThis server provides...',
|
|
213
|
-
}],
|
|
214
|
-
}));
|
|
215
|
-
```
|
|
216
|
-
|
|
217
|
-
### Dynamic Resource
|
|
218
|
-
|
|
219
|
-
```typescript
|
|
220
|
-
server.resource('Config', 'resource://config', async () => {
|
|
221
|
-
const config = await loadConfig();
|
|
222
|
-
return {
|
|
223
|
-
contents: [{
|
|
224
|
-
uri: 'resource://config',
|
|
225
|
-
mimeType: 'application/json',
|
|
226
|
-
text: JSON.stringify(config, null, 2),
|
|
227
|
-
}],
|
|
228
|
-
};
|
|
229
|
-
});
|
|
230
|
-
```
|
|
231
|
-
|
|
232
|
-
### File Resource
|
|
233
|
-
|
|
234
|
-
```typescript
|
|
235
|
-
import { readFile } from 'fs/promises';
|
|
236
|
-
|
|
237
|
-
server.resource('Schema', 'file://schema.json', async () => {
|
|
238
|
-
const content = await readFile('./schema.json', 'utf-8');
|
|
239
|
-
return {
|
|
240
|
-
contents: [{
|
|
241
|
-
uri: 'file://schema.json',
|
|
242
|
-
mimeType: 'application/json',
|
|
243
|
-
text: content,
|
|
244
|
-
}],
|
|
245
|
-
};
|
|
246
|
-
});
|
|
247
|
-
```
|
|
248
|
-
|
|
249
|
-
### Resource with Parameters
|
|
250
|
-
|
|
251
|
-
```typescript
|
|
252
|
-
// Register a resource template
|
|
253
|
-
server.resource(
|
|
254
|
-
'User Profile',
|
|
255
|
-
'resource://user/{id}',
|
|
256
|
-
async (uri) => {
|
|
257
|
-
const id = uri.pathname.split('/').pop();
|
|
258
|
-
const user = await getUser(id);
|
|
259
|
-
return {
|
|
260
|
-
contents: [{
|
|
261
|
-
uri: uri.href,
|
|
262
|
-
mimeType: 'application/json',
|
|
263
|
-
text: JSON.stringify(user),
|
|
264
|
-
}],
|
|
265
|
-
};
|
|
266
|
-
}
|
|
267
|
-
);
|
|
268
|
-
```
|
|
269
|
-
|
|
270
|
-
### Binary Resource
|
|
271
|
-
|
|
272
|
-
```typescript
|
|
273
|
-
server.resource('Logo', 'resource://logo.png', async () => {
|
|
274
|
-
const buffer = await readFile('./logo.png');
|
|
275
|
-
return {
|
|
276
|
-
contents: [{
|
|
277
|
-
uri: 'resource://logo.png',
|
|
278
|
-
mimeType: 'image/png',
|
|
279
|
-
blob: buffer.toString('base64'),
|
|
280
|
-
}],
|
|
281
|
-
};
|
|
282
|
-
});
|
|
283
|
-
```
|
|
284
|
-
|
|
285
|
-
## Prompt Patterns
|
|
286
|
-
|
|
287
|
-
### Simple Prompt
|
|
288
|
-
|
|
289
|
-
```typescript
|
|
290
|
-
server.prompt(
|
|
291
|
-
'summarize',
|
|
292
|
-
'Summarize the given text',
|
|
293
|
-
{
|
|
294
|
-
text: z.string().describe('Text to summarize'),
|
|
295
|
-
},
|
|
296
|
-
async (args) => ({
|
|
297
|
-
description: 'Text summarization prompt',
|
|
298
|
-
messages: [{
|
|
299
|
-
role: 'user',
|
|
300
|
-
content: { type: 'text', text: `Please summarize:\n\n${args.text}` },
|
|
301
|
-
}],
|
|
302
|
-
})
|
|
303
|
-
);
|
|
304
|
-
```
|
|
305
|
-
|
|
306
|
-
### Prompt with System Message
|
|
307
|
-
|
|
308
|
-
```typescript
|
|
309
|
-
server.prompt(
|
|
310
|
-
'code_review',
|
|
311
|
-
'Review code for issues',
|
|
312
|
-
{
|
|
313
|
-
code: z.string().describe('Code to review'),
|
|
314
|
-
language: z.string().optional().describe('Programming language'),
|
|
315
|
-
},
|
|
316
|
-
async (args) => ({
|
|
317
|
-
description: 'Code review prompt',
|
|
318
|
-
messages: [
|
|
319
|
-
{
|
|
320
|
-
role: 'system',
|
|
321
|
-
content: {
|
|
322
|
-
type: 'text',
|
|
323
|
-
text: 'You are an expert code reviewer. Focus on security, performance, and maintainability.',
|
|
324
|
-
},
|
|
325
|
-
},
|
|
326
|
-
{
|
|
327
|
-
role: 'user',
|
|
328
|
-
content: {
|
|
329
|
-
type: 'text',
|
|
330
|
-
text: `Review this ${args.language || 'code'}:\n\n\`\`\`${args.language || ''}\n${args.code}\n\`\`\``,
|
|
331
|
-
},
|
|
332
|
-
},
|
|
333
|
-
],
|
|
334
|
-
})
|
|
335
|
-
);
|
|
336
|
-
```
|
|
337
|
-
|
|
338
|
-
### Prompt with Context
|
|
339
|
-
|
|
340
|
-
```typescript
|
|
341
|
-
server.prompt(
|
|
342
|
-
'analyze_with_context',
|
|
343
|
-
'Analyze data with additional context',
|
|
344
|
-
{
|
|
345
|
-
data: z.string().describe('Data to analyze'),
|
|
346
|
-
context: z.string().optional().describe('Additional context'),
|
|
347
|
-
focus: z.enum(['performance', 'security', 'quality']).describe('Analysis focus'),
|
|
348
|
-
},
|
|
349
|
-
async (args) => ({
|
|
350
|
-
description: `${args.focus} analysis prompt`,
|
|
351
|
-
messages: [
|
|
352
|
-
...(args.context ? [{
|
|
353
|
-
role: 'system' as const,
|
|
354
|
-
content: { type: 'text' as const, text: args.context },
|
|
355
|
-
}] : []),
|
|
356
|
-
{
|
|
357
|
-
role: 'user',
|
|
358
|
-
content: {
|
|
359
|
-
type: 'text',
|
|
360
|
-
text: `Analyze the following with focus on ${args.focus}:\n\n${args.data}`,
|
|
361
|
-
},
|
|
362
|
-
},
|
|
363
|
-
],
|
|
364
|
-
})
|
|
365
|
-
);
|
|
366
|
-
```
|
|
367
|
-
|
|
368
|
-
## Tool Naming Conventions
|
|
369
|
-
|
|
370
|
-
AI assistants select tools based on names and descriptions. Follow these conventions for optimal discoverability:
|
|
371
|
-
|
|
372
|
-
### Naming Pattern: `verb_noun`
|
|
373
|
-
|
|
374
|
-
```text
|
|
375
|
-
Format: {action}_{resource}
|
|
376
|
-
Examples: get_user, list_items, create_order, delete_file, search_documents
|
|
377
|
-
```
|
|
378
|
-
|
|
379
|
-
| Verb | When to Use | Examples |
|
|
380
|
-
|------|-------------|----------|
|
|
381
|
-
| `get` | Retrieve single item by ID | `get_user`, `get_order`, `get_config` |
|
|
382
|
-
| `list` | Retrieve multiple items | `list_users`, `list_orders`, `list_files` |
|
|
383
|
-
| `search` | Query with filters | `search_documents`, `search_logs` |
|
|
384
|
-
| `create` | Create new resource | `create_user`, `create_project` |
|
|
385
|
-
| `update` | Modify existing resource | `update_user`, `update_settings` |
|
|
386
|
-
| `delete` | Remove resource | `delete_user`, `delete_file` |
|
|
387
|
-
| `validate` | Check validity | `validate_email`, `validate_config` |
|
|
388
|
-
| `convert` | Transform format | `convert_currency`, `convert_image` |
|
|
389
|
-
| `send` | Transmit data | `send_email`, `send_notification` |
|
|
390
|
-
| `run` | Execute process | `run_build`, `run_tests` |
|
|
391
|
-
|
|
392
|
-
### Naming Anti-Patterns
|
|
393
|
-
|
|
394
|
-
```typescript
|
|
395
|
-
// BAD: Vague or generic names
|
|
396
|
-
'do_thing' // What thing?
|
|
397
|
-
'process' // Process what?
|
|
398
|
-
'handle_data' // Too generic
|
|
399
|
-
'execute' // Execute what?
|
|
400
|
-
|
|
401
|
-
// BAD: Inconsistent casing
|
|
402
|
-
'getUser' // Use snake_case, not camelCase
|
|
403
|
-
'Get_User' // Inconsistent capitalization
|
|
404
|
-
'GET_USER' // All caps
|
|
405
|
-
|
|
406
|
-
// BAD: Redundant prefixes
|
|
407
|
-
'tool_get_user' // "tool_" prefix is redundant
|
|
408
|
-
'mcp_list_items' // "mcp_" prefix is redundant
|
|
409
|
-
|
|
410
|
-
// GOOD: Clear, consistent, descriptive
|
|
411
|
-
'get_user'
|
|
412
|
-
'list_items'
|
|
413
|
-
'search_documents'
|
|
414
|
-
'create_order'
|
|
415
|
-
```
|
|
416
|
-
|
|
417
|
-
### Compound Actions
|
|
418
|
-
|
|
419
|
-
For complex operations, use descriptive compound names:
|
|
420
|
-
|
|
421
|
-
```typescript
|
|
422
|
-
// Multi-step operations
|
|
423
|
-
'get_user_with_orders' // Returns user + their orders
|
|
424
|
-
'list_active_sessions' // Filtered list
|
|
425
|
-
'search_and_replace' // Combined action
|
|
426
|
-
|
|
427
|
-
// Domain-specific operations
|
|
428
|
-
'deploy_to_production'
|
|
429
|
-
'sync_inventory'
|
|
430
|
-
'generate_report'
|
|
431
|
-
'export_to_csv'
|
|
432
|
-
```
|
|
433
|
-
|
|
434
|
-
## Tool Description Guidelines
|
|
435
|
-
|
|
436
|
-
Descriptions are critical for AI assistants to understand when and how to use tools.
|
|
437
|
-
|
|
438
|
-
### Description Structure
|
|
439
|
-
|
|
440
|
-
Every tool description should answer:
|
|
441
|
-
|
|
442
|
-
1. **What** - What does this tool do?
|
|
443
|
-
2. **When** - When should an AI use this tool?
|
|
444
|
-
3. **Input** - What inputs are required/optional?
|
|
445
|
-
4. **Output** - What does it return?
|
|
446
|
-
5. **Side Effects** - Does it modify state?
|
|
447
|
-
|
|
448
|
-
### Description Template
|
|
449
|
-
|
|
450
|
-
```typescript
|
|
451
|
-
server.tool(
|
|
452
|
-
'create_user',
|
|
453
|
-
// Description should be 1-2 sentences covering purpose and key behavior
|
|
454
|
-
'Creates a new user account with the specified details. Returns the created user object with generated ID.',
|
|
455
|
-
{
|
|
456
|
-
email: z.string().email().describe('User email address (must be unique)'),
|
|
457
|
-
name: z.string().describe('Full name of the user'),
|
|
458
|
-
role: z.enum(['admin', 'user', 'guest']).optional().describe('User role (default: user)'),
|
|
459
|
-
},
|
|
460
|
-
async (args) => { /* ... */ }
|
|
461
|
-
);
|
|
462
|
-
```
|
|
463
|
-
|
|
464
|
-
### Good vs Bad Descriptions
|
|
465
|
-
|
|
466
|
-
```typescript
|
|
467
|
-
// BAD: Too vague
|
|
468
|
-
server.tool('get_data', 'Gets data', { /* ... */ });
|
|
469
|
-
|
|
470
|
-
// BAD: Missing key information
|
|
471
|
-
server.tool('delete_user', 'Deletes a user', { /* ... */ });
|
|
472
|
-
// Missing: What happens to user's data? Is this reversible?
|
|
473
|
-
|
|
474
|
-
// BAD: Too technical, not AI-friendly
|
|
475
|
-
server.tool('query_db', 'Executes SELECT * FROM users WHERE id = ?', { /* ... */ });
|
|
476
|
-
|
|
477
|
-
// GOOD: Clear purpose, behavior, and constraints
|
|
478
|
-
server.tool(
|
|
479
|
-
'get_user',
|
|
480
|
-
'Retrieves a user by their unique ID. Returns user profile including name, email, and role. Returns null if user not found.',
|
|
481
|
-
{ id: z.string().uuid().describe('Unique user identifier (UUID format)') }
|
|
482
|
-
);
|
|
483
|
-
|
|
484
|
-
// GOOD: Explains side effects
|
|
485
|
-
server.tool(
|
|
486
|
-
'delete_user',
|
|
487
|
-
'Permanently deletes a user account and all associated data. This action cannot be undone. Requires admin privileges.',
|
|
488
|
-
{ id: z.string().uuid().describe('User ID to delete') }
|
|
489
|
-
);
|
|
490
|
-
|
|
491
|
-
// GOOD: Explains when to use
|
|
492
|
-
server.tool(
|
|
493
|
-
'search_documents',
|
|
494
|
-
'Full-text search across all documents. Use this when looking for documents by content rather than by ID. Supports wildcards and phrase matching.',
|
|
495
|
-
{
|
|
496
|
-
query: z.string().describe('Search query (supports * wildcards and "exact phrases")'),
|
|
497
|
-
limit: z.number().optional().default(20).describe('Maximum results to return'),
|
|
498
|
-
}
|
|
499
|
-
);
|
|
500
|
-
```
|
|
501
|
-
|
|
502
|
-
### Parameter Description Best Practices
|
|
503
|
-
|
|
504
|
-
```typescript
|
|
505
|
-
// BAD: No description
|
|
506
|
-
query: z.string()
|
|
507
|
-
|
|
508
|
-
// BAD: Redundant description
|
|
509
|
-
query: z.string().describe('Query')
|
|
510
|
-
|
|
511
|
-
// BAD: Missing constraints
|
|
512
|
-
limit: z.number().describe('Limit')
|
|
513
|
-
|
|
514
|
-
// GOOD: Explains purpose and constraints
|
|
515
|
-
query: z.string().describe('Search query - supports wildcards (*) and exact phrases ("...")')
|
|
516
|
-
|
|
517
|
-
// GOOD: Includes valid values
|
|
518
|
-
status: z.enum(['pending', 'active', 'completed']).describe('Filter by status')
|
|
519
|
-
|
|
520
|
-
// GOOD: Explains defaults and ranges
|
|
521
|
-
limit: z.number().min(1).max(100).optional().default(20)
|
|
522
|
-
.describe('Results per page (1-100, default: 20)')
|
|
523
|
-
|
|
524
|
-
// GOOD: Explains format requirements
|
|
525
|
-
date: z.string().describe('Date in ISO 8601 format (YYYY-MM-DD)')
|
|
526
|
-
|
|
527
|
-
// GOOD: Explains relationships
|
|
528
|
-
user_id: z.string().uuid().describe('ID of the user who owns this resource')
|
|
529
|
-
```
|
|
530
|
-
|
|
531
|
-
### Indicating Side Effects
|
|
532
|
-
|
|
533
|
-
Always document if a tool modifies state:
|
|
534
|
-
|
|
535
|
-
```typescript
|
|
536
|
-
// Read-only tool - no side effects
|
|
537
|
-
server.tool(
|
|
538
|
-
'get_user',
|
|
539
|
-
'Retrieves user details. Read-only operation.',
|
|
540
|
-
{ /* ... */ }
|
|
541
|
-
);
|
|
542
|
-
|
|
543
|
-
// Write operation - has side effects
|
|
544
|
-
server.tool(
|
|
545
|
-
'update_user',
|
|
546
|
-
'Updates user profile fields. Modifies the user record in the database.',
|
|
547
|
-
{ /* ... */ }
|
|
548
|
-
);
|
|
549
|
-
|
|
550
|
-
// Destructive operation - irreversible
|
|
551
|
-
server.tool(
|
|
552
|
-
'delete_user',
|
|
553
|
-
'Permanently deletes user and all data. IRREVERSIBLE. Requires confirmation.',
|
|
554
|
-
{ /* ... */ }
|
|
555
|
-
);
|
|
556
|
-
|
|
557
|
-
// External side effects
|
|
558
|
-
server.tool(
|
|
559
|
-
'send_email',
|
|
560
|
-
'Sends an email to the specified recipient. Cannot be undone once sent.',
|
|
561
|
-
{ /* ... */ }
|
|
562
|
-
);
|
|
563
|
-
```
|
|
564
|
-
|
|
565
|
-
## Best Practices
|
|
566
|
-
|
|
567
|
-
### 1. Always Describe Parameters
|
|
568
|
-
|
|
569
|
-
```typescript
|
|
570
|
-
// Good
|
|
571
|
-
query: z.string().describe('Search query to find matching items')
|
|
572
|
-
|
|
573
|
-
// Bad
|
|
574
|
-
query: z.string()
|
|
575
|
-
```
|
|
576
|
-
|
|
577
|
-
### 2. Use Specific Types
|
|
578
|
-
|
|
579
|
-
```typescript
|
|
580
|
-
// Good
|
|
581
|
-
email: z.string().email().describe('User email address')
|
|
582
|
-
url: z.string().url().describe('Webhook URL')
|
|
583
|
-
count: z.number().int().positive().describe('Number of items')
|
|
584
|
-
|
|
585
|
-
// Bad
|
|
586
|
-
email: z.string()
|
|
587
|
-
url: z.string()
|
|
588
|
-
count: z.number()
|
|
589
|
-
```
|
|
590
|
-
|
|
591
|
-
### 3. Provide Defaults for Optional Parameters
|
|
592
|
-
|
|
593
|
-
```typescript
|
|
594
|
-
// Good
|
|
595
|
-
limit: z.number().optional().default(20).describe('Max results (default: 20)')
|
|
596
|
-
|
|
597
|
-
// Okay but less clear
|
|
598
|
-
limit: z.number().optional().describe('Max results')
|
|
599
|
-
```
|
|
600
|
-
|
|
601
|
-
### 4. Return Structured JSON
|
|
602
|
-
|
|
603
|
-
```typescript
|
|
604
|
-
// Good - structured, parseable
|
|
605
|
-
return {
|
|
606
|
-
content: [{
|
|
607
|
-
type: 'text',
|
|
608
|
-
text: JSON.stringify({ success: true, data: result }, null, 2),
|
|
609
|
-
}],
|
|
610
|
-
};
|
|
611
|
-
|
|
612
|
-
// Bad - unstructured text
|
|
613
|
-
return {
|
|
614
|
-
content: [{ type: 'text', text: `Success! Got ${result.length} items` }],
|
|
615
|
-
};
|
|
616
|
-
```
|
|
617
|
-
|
|
618
|
-
### 5. Handle Errors Gracefully
|
|
619
|
-
|
|
620
|
-
```typescript
|
|
621
|
-
// Good - structured error
|
|
622
|
-
return {
|
|
623
|
-
content: [{
|
|
624
|
-
type: 'text',
|
|
625
|
-
text: JSON.stringify({ error: true, code: 'NOT_FOUND', message: 'Item not found' }),
|
|
626
|
-
}],
|
|
627
|
-
isError: true,
|
|
628
|
-
};
|
|
629
|
-
|
|
630
|
-
// Bad - throws exception
|
|
631
|
-
throw new Error('Item not found');
|
|
632
|
-
```
|