@serpstat/serpstat-mcp-server 1.0.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/.env.example +10 -0
- package/LICENSE +21 -0
- package/README.md +288 -0
- package/dist/handlers/backlinks_tools.d.ts +11 -0
- package/dist/handlers/backlinks_tools.d.ts.map +1 -0
- package/dist/handlers/backlinks_tools.js +59 -0
- package/dist/handlers/backlinks_tools.js.map +1 -0
- package/dist/handlers/base.d.ts +10 -0
- package/dist/handlers/base.d.ts.map +1 -0
- package/dist/handlers/base.js +26 -0
- package/dist/handlers/base.js.map +1 -0
- package/dist/handlers/domain.d.ts +11 -0
- package/dist/handlers/domain.d.ts.map +1 -0
- package/dist/handlers/domain.js +75 -0
- package/dist/handlers/domain.js.map +1 -0
- package/dist/handlers/domain_tools.d.ts +51 -0
- package/dist/handlers/domain_tools.d.ts.map +1 -0
- package/dist/handlers/domain_tools.js +538 -0
- package/dist/handlers/domain_tools.js.map +1 -0
- package/dist/handlers/keyword_tools.d.ts +19 -0
- package/dist/handlers/keyword_tools.d.ts.map +1 -0
- package/dist/handlers/keyword_tools.js +251 -0
- package/dist/handlers/keyword_tools.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +31 -0
- package/dist/index.js.map +1 -0
- package/dist/server.d.ts +9 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +76 -0
- package/dist/server.js.map +1 -0
- package/dist/services/backlinks_tools.d.ts +7 -0
- package/dist/services/backlinks_tools.d.ts.map +1 -0
- package/dist/services/backlinks_tools.js +25 -0
- package/dist/services/backlinks_tools.js.map +1 -0
- package/dist/services/base.d.ts +13 -0
- package/dist/services/base.d.ts.map +1 -0
- package/dist/services/base.js +83 -0
- package/dist/services/base.js.map +1 -0
- package/dist/services/domain.d.ts +7 -0
- package/dist/services/domain.d.ts.map +1 -0
- package/dist/services/domain.js +26 -0
- package/dist/services/domain.js.map +1 -0
- package/dist/services/domain_tools.d.ts +13 -0
- package/dist/services/domain_tools.d.ts.map +1 -0
- package/dist/services/domain_tools.js +111 -0
- package/dist/services/domain_tools.js.map +1 -0
- package/dist/services/keyword_tools.d.ts +8 -0
- package/dist/services/keyword_tools.d.ts.map +1 -0
- package/dist/services/keyword_tools.js +43 -0
- package/dist/services/keyword_tools.js.map +1 -0
- package/dist/types/mcp.d.ts +12 -0
- package/dist/types/mcp.d.ts.map +1 -0
- package/dist/types/mcp.js +3 -0
- package/dist/types/mcp.js.map +1 -0
- package/dist/types/serpstat.d.ts +237 -0
- package/dist/types/serpstat.d.ts.map +1 -0
- package/dist/types/serpstat.js +24 -0
- package/dist/types/serpstat.js.map +1 -0
- package/dist/utils/config.d.ts +24 -0
- package/dist/utils/config.d.ts.map +1 -0
- package/dist/utils/config.js +21 -0
- package/dist/utils/config.js.map +1 -0
- package/dist/utils/logger.d.ts +3 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +19 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/validation.d.ts +952 -0
- package/dist/utils/validation.d.ts.map +1 -0
- package/dist/utils/validation.js +257 -0
- package/dist/utils/validation.js.map +1 -0
- package/package.json +46 -0
package/.env.example
ADDED
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2025 Serpstat Global Limited
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
package/README.md
ADDED
|
@@ -0,0 +1,288 @@
|
|
|
1
|
+
# Serpstat MCP Server
|
|
2
|
+
|
|
3
|
+
[](https://badge.fury.io/js/serpstat-mcp-server)
|
|
4
|
+
[](https://opensource.org/licenses/MIT)
|
|
5
|
+
|
|
6
|
+
Server for integrating Serpstat with Model Context Protocol (MCP).
|
|
7
|
+
|
|
8
|
+
## Description
|
|
9
|
+
|
|
10
|
+
This project implements a TypeScript server that provides an API interface for working with Serpstat tools via the MCP protocol. The server supports request handling, parameter validation, logging, and working with multiple tools.
|
|
11
|
+
|
|
12
|
+
## Features
|
|
13
|
+
|
|
14
|
+
- Retrieve domain information via Serpstat API
|
|
15
|
+
- Input parameter validation using Zod
|
|
16
|
+
- Event logging with Winston
|
|
17
|
+
- Flexible configuration via environment variables
|
|
18
|
+
- Jest tests for parameter and logic validation
|
|
19
|
+
|
|
20
|
+
## Requirements
|
|
21
|
+
|
|
22
|
+
- Node.js 18.0.0 or higher
|
|
23
|
+
- Valid Serpstat API token
|
|
24
|
+
|
|
25
|
+
## Installation
|
|
26
|
+
|
|
27
|
+
### Global Installation (Recommended)
|
|
28
|
+
|
|
29
|
+
```bash
|
|
30
|
+
npm install -g @serpstat/serpstat-mcp-server
|
|
31
|
+
```
|
|
32
|
+
|
|
33
|
+
### Local Installation
|
|
34
|
+
|
|
35
|
+
```bash
|
|
36
|
+
npm install @serpstat/serpstat-mcp-server
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
## Configuration
|
|
40
|
+
|
|
41
|
+
### Environment Variables
|
|
42
|
+
|
|
43
|
+
Set the following environment variables (can be in .env file):
|
|
44
|
+
|
|
45
|
+
- `SERPSTAT_API_TOKEN` — Your Serpstat API token (required)
|
|
46
|
+
- `SERPSTAT_API_URL` — Serpstat API URL (default: https://api.serpstat.com/v4)
|
|
47
|
+
- `LOG_LEVEL` — Logging level: error, warn, info, debug (default: info)
|
|
48
|
+
|
|
49
|
+
### Claude Desktop Configuration
|
|
50
|
+
|
|
51
|
+
Add to your Claude Desktop config file:
|
|
52
|
+
|
|
53
|
+
**MacOS**: `~/Library/Application Support/Claude/claude_desktop_config.json`
|
|
54
|
+
**Windows**: `%APPDATA%/Claude/claude_desktop_config.json`
|
|
55
|
+
|
|
56
|
+
```json
|
|
57
|
+
{
|
|
58
|
+
"mcpServers": {
|
|
59
|
+
"serpstat": {
|
|
60
|
+
"command": "npx",
|
|
61
|
+
"args": ["-y", "@serpstat/serpstat-mcp-server"],
|
|
62
|
+
"env": {
|
|
63
|
+
"SERPSTAT_API_TOKEN": "XXXXXXXXXXXPLACETOKENHEREXXXXXXXXXXXXX",
|
|
64
|
+
"LANG": "en_US.UTF-8",
|
|
65
|
+
"LC_ALL": "en_US.UTF-8"
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
```
|
|
71
|
+
|
|
72
|
+
For local development, use the full path:
|
|
73
|
+
|
|
74
|
+
```json
|
|
75
|
+
{
|
|
76
|
+
"mcpServers": {
|
|
77
|
+
"serpstat": {
|
|
78
|
+
"command": "node",
|
|
79
|
+
"args": ["/path/to/node_modules/serpstat-mcp-server/dist/index.js"],
|
|
80
|
+
"env": {
|
|
81
|
+
"SERPSTAT_API_TOKEN": "your_serpstat_api_token_here"
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
```
|
|
87
|
+
|
|
88
|
+
## Usage Examples
|
|
89
|
+
|
|
90
|
+
After installation and configuration in Claude Desktop, you can ask Claude:
|
|
91
|
+
|
|
92
|
+
- "Show me domain info for example.com"
|
|
93
|
+
- "Find competitors for my-site.com in Google US"
|
|
94
|
+
- "Get top 50 keywords that example.com ranks for"
|
|
95
|
+
- "Analyze backlinks summary for domain.com"
|
|
96
|
+
- "Find related keywords to 'digital marketing'"
|
|
97
|
+
- "Get unique keywords for domain1.com vs domain2.com"
|
|
98
|
+
|
|
99
|
+
## Development
|
|
100
|
+
|
|
101
|
+
### Getting Started
|
|
102
|
+
|
|
103
|
+
1. Clone the repository:
|
|
104
|
+
```bash
|
|
105
|
+
git clone <repository-url>
|
|
106
|
+
cd serpstat-mcp-server
|
|
107
|
+
```
|
|
108
|
+
|
|
109
|
+
2. Install dependencies:
|
|
110
|
+
```bash
|
|
111
|
+
npm install
|
|
112
|
+
```
|
|
113
|
+
|
|
114
|
+
3. Set environment variables:
|
|
115
|
+
```bash
|
|
116
|
+
cp .env.example .env
|
|
117
|
+
# Edit .env with your Serpstat API token
|
|
118
|
+
```
|
|
119
|
+
|
|
120
|
+
4. Build the project:
|
|
121
|
+
```bash
|
|
122
|
+
npm run build
|
|
123
|
+
```
|
|
124
|
+
|
|
125
|
+
5. Start the server:
|
|
126
|
+
```bash
|
|
127
|
+
npm start
|
|
128
|
+
```
|
|
129
|
+
|
|
130
|
+
6. For development mode (auto-reload):
|
|
131
|
+
```bash
|
|
132
|
+
npm run dev
|
|
133
|
+
```
|
|
134
|
+
|
|
135
|
+
### Testing
|
|
136
|
+
|
|
137
|
+
To run tests:
|
|
138
|
+
```bash
|
|
139
|
+
npm test
|
|
140
|
+
```
|
|
141
|
+
|
|
142
|
+
### Scripts
|
|
143
|
+
|
|
144
|
+
- `npm run build` — Compile TypeScript sources to JavaScript (output in `dist/`)
|
|
145
|
+
- `npm start` — Run the compiled server from `dist/`
|
|
146
|
+
- `npm run dev` — Run the server in development mode with hot-reload
|
|
147
|
+
- `npm test` — Run all tests
|
|
148
|
+
- `npm run lint` — Run linting
|
|
149
|
+
- `npm run clean` — Clean build directory
|
|
150
|
+
|
|
151
|
+
## Project Structure
|
|
152
|
+
|
|
153
|
+
```
|
|
154
|
+
serpstat-mcp-server/
|
|
155
|
+
├── src/
|
|
156
|
+
│ ├── index.ts # Entry point
|
|
157
|
+
│ ├── server.ts # Main MCP server
|
|
158
|
+
│ ├── handlers/ # Tool handlers
|
|
159
|
+
│ ├── services/ # Services for Serpstat API
|
|
160
|
+
│ ├── types/ # Data types
|
|
161
|
+
│ ├── utils/ # Utilities (config, logger, validation)
|
|
162
|
+
│ └── __tests__/ # Tests
|
|
163
|
+
├── dist/ # Compiled JavaScript (after build)
|
|
164
|
+
├── package.json
|
|
165
|
+
├── tsconfig.json
|
|
166
|
+
├── README.md
|
|
167
|
+
└── .env.example
|
|
168
|
+
```
|
|
169
|
+
|
|
170
|
+
## MCP Tools
|
|
171
|
+
|
|
172
|
+
| Tool Name | Description | Key Parameters |
|
|
173
|
+
|---------------------------|-----------------------------------------------------------------------------------------------|-----------------------------|
|
|
174
|
+
| get_domains_info | Get SEO information for multiple domains | domains, se, filters |
|
|
175
|
+
| get_domain_competitors | Get list of competitor domains | domain, se, size, filters |
|
|
176
|
+
| get_domain_keywords | Get keywords that domain ranks for | domain, se, page, size |
|
|
177
|
+
| get_domain_urls | Get URLs within a domain and their keyword counts | domain, se, page, size |
|
|
178
|
+
| get_domain_regions_count | Get keyword count by region for a domain | domain, sort, order |
|
|
179
|
+
| get_domain_uniq_keywords | Get unique keywords for two domains not ranked by a third domain | se, domains, minusDomain |
|
|
180
|
+
| get_keywords | Get related organic keywords for a given keyword | keyword, se, filters |
|
|
181
|
+
| get_backlinks_summary | Get comprehensive backlinks summary with referring domains, quality metrics, and changes | domain, subdomain |
|
|
182
|
+
| get_related_keywords | Get semantically related keywords with frequency, CPC, competition, and difficulty data | keyword, se, filters, sort |
|
|
183
|
+
|
|
184
|
+
### Search Engines (se parameter)
|
|
185
|
+
|
|
186
|
+
Common search engine codes:
|
|
187
|
+
- `g_us` - Google US
|
|
188
|
+
- `g_uk` - Google UK
|
|
189
|
+
- `g_ca` - Google Canada
|
|
190
|
+
- `g_au` - Google Australia
|
|
191
|
+
- `g_de` - Google Germany
|
|
192
|
+
- `g_fr` - Google France
|
|
193
|
+
- `g_es` - Google Spain
|
|
194
|
+
- `g_it` - Google Italy
|
|
195
|
+
|
|
196
|
+
|
|
197
|
+
## Troubleshooting
|
|
198
|
+
|
|
199
|
+
### Common Issues
|
|
200
|
+
|
|
201
|
+
**"Command not found: serpstat-mcp-server"**
|
|
202
|
+
- Make sure you installed the package globally with `-g` flag
|
|
203
|
+
- Verify your PATH includes npm global binaries: `npm config get prefix`
|
|
204
|
+
- Try reinstalling: `npm uninstall -g serpstat-mcp-server && npm install -g serpstat-mcp-server`
|
|
205
|
+
|
|
206
|
+
**"API token error" or "Unauthorized"**
|
|
207
|
+
- Check that `SERPSTAT_API_TOKEN` is set correctly in your environment
|
|
208
|
+
- Verify your token is valid and active in your Serpstat account
|
|
209
|
+
- Ensure your token has sufficient API credits and permissions
|
|
210
|
+
|
|
211
|
+
**"Module not found" errors**
|
|
212
|
+
- Make sure all dependencies are installed: `npm install`
|
|
213
|
+
- Try rebuilding: `npm run clean && npm run build`
|
|
214
|
+
|
|
215
|
+
**Claude Desktop doesn't recognize the server**
|
|
216
|
+
- Restart Claude Desktop after configuration changes
|
|
217
|
+
- Check the config file path and JSON syntax
|
|
218
|
+
- Verify the server starts correctly: run `serpstat-mcp-server` in terminal
|
|
219
|
+
|
|
220
|
+
### Debug Mode
|
|
221
|
+
|
|
222
|
+
Enable debug logging by setting:
|
|
223
|
+
```bash
|
|
224
|
+
export LOG_LEVEL=debug
|
|
225
|
+
```
|
|
226
|
+
|
|
227
|
+
Or in your Claude Desktop config for linux:
|
|
228
|
+
```json
|
|
229
|
+
{
|
|
230
|
+
"mcpServers": {
|
|
231
|
+
"serpstat": {
|
|
232
|
+
"command": "npx",
|
|
233
|
+
"args": ["-y", "@serpstat/serpstat-mcp-server"],
|
|
234
|
+
"env": {
|
|
235
|
+
"SERPSTAT_API_TOKEN": "XXXXXXXXXXXPLACETOKENHEREXXXXXXXXXXXXX",
|
|
236
|
+
"LANG": "en_US.UTF-8",
|
|
237
|
+
"LC_ALL": "en_US.UTF-8"
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
```
|
|
243
|
+
|
|
244
|
+
## API Rate Limits
|
|
245
|
+
|
|
246
|
+
Serpstat API has rate limits. The server includes:
|
|
247
|
+
- Automatic retry logic with exponential backoff
|
|
248
|
+
- Request queuing to prevent rate limit violations
|
|
249
|
+
- Detailed error messages for quota exceeded scenarios
|
|
250
|
+
|
|
251
|
+
## Contributing
|
|
252
|
+
|
|
253
|
+
We welcome contributions! Please follow these steps:
|
|
254
|
+
|
|
255
|
+
1. Fork the repository
|
|
256
|
+
2. Create a feature branch: `git checkout -b feature/amazing-feature`
|
|
257
|
+
3. Make your changes
|
|
258
|
+
4. Add tests for new functionality
|
|
259
|
+
5. Ensure tests pass: `npm test`
|
|
260
|
+
6. Commit your changes: `git commit -m 'Add amazing feature'`
|
|
261
|
+
7. Push to the branch: `git push origin feature/amazing-feature`
|
|
262
|
+
8. Submit a pull request
|
|
263
|
+
|
|
264
|
+
### Development Guidelines
|
|
265
|
+
|
|
266
|
+
- Follow existing code style and TypeScript conventions
|
|
267
|
+
- Add tests for new features
|
|
268
|
+
- Update documentation as needed
|
|
269
|
+
- Use conventional commit messages
|
|
270
|
+
|
|
271
|
+
## Changelog
|
|
272
|
+
|
|
273
|
+
See [CHANGELOG.md](CHANGELOG.md) for details about changes in each version.
|
|
274
|
+
|
|
275
|
+
## Support
|
|
276
|
+
|
|
277
|
+
- 📖 [Serpstat API Documentation](https://serpstat.com/api/)
|
|
278
|
+
- 🐛 [Report Issues](https://github.com/your-username/serpstat-mcp-server/issues)
|
|
279
|
+
- 💬 [Discussions](https://github.com/your-username/serpstat-mcp-server/discussions)
|
|
280
|
+
|
|
281
|
+
## License
|
|
282
|
+
|
|
283
|
+
This project is licensed under the MIT License - see the [LICENSE](LICENSE) file for details.
|
|
284
|
+
|
|
285
|
+
## Acknowledgments
|
|
286
|
+
|
|
287
|
+
- [Model Context Protocol](https://github.com/modelcontextprotocol) by Anthropic
|
|
288
|
+
- [Serpstat API](https://serpstat.com/api/) for SEO data services
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { BaseHandler } from './base.js';
|
|
2
|
+
import { MCPToolCall, MCPToolResponse } from '../types/mcp.js';
|
|
3
|
+
export declare class BacklinksSummaryHandler extends BaseHandler {
|
|
4
|
+
private backlinksService;
|
|
5
|
+
constructor();
|
|
6
|
+
getName(): string;
|
|
7
|
+
getDescription(): string;
|
|
8
|
+
getInputSchema(): object;
|
|
9
|
+
handle(call: MCPToolCall): Promise<MCPToolResponse>;
|
|
10
|
+
}
|
|
11
|
+
//# sourceMappingURL=backlinks_tools.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"backlinks_tools.d.ts","sourceRoot":"","sources":["../../src/handlers/backlinks_tools.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AAExC,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAK/D,qBAAa,uBAAwB,SAAQ,WAAW;IACpD,OAAO,CAAC,gBAAgB,CAAmB;;IAQ3C,OAAO,IAAI,MAAM;IAIjB,cAAc,IAAI,MAAM;IAIxB,cAAc,IAAI,MAAM;IAuBlB,MAAM,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,eAAe,CAAC;CAY5D"}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.BacklinksSummaryHandler = void 0;
|
|
4
|
+
const base_js_1 = require("./base.js");
|
|
5
|
+
const backlinks_tools_js_1 = require("../services/backlinks_tools.js");
|
|
6
|
+
const validation_js_1 = require("../utils/validation.js");
|
|
7
|
+
const config_js_1 = require("../utils/config.js");
|
|
8
|
+
const zod_1 = require("zod");
|
|
9
|
+
class BacklinksSummaryHandler extends base_js_1.BaseHandler {
|
|
10
|
+
backlinksService;
|
|
11
|
+
constructor() {
|
|
12
|
+
super();
|
|
13
|
+
const config = (0, config_js_1.loadConfig)();
|
|
14
|
+
this.backlinksService = new backlinks_tools_js_1.BacklinksService(config);
|
|
15
|
+
}
|
|
16
|
+
getName() {
|
|
17
|
+
return 'get_backlinks_summary';
|
|
18
|
+
}
|
|
19
|
+
getDescription() {
|
|
20
|
+
return 'Get comprehensive backlinks summary using Serpstat API. Returns referring domains, backlinks count, link types, quality metrics and recent changes for domain or subdomain.';
|
|
21
|
+
}
|
|
22
|
+
getInputSchema() {
|
|
23
|
+
return {
|
|
24
|
+
type: "object",
|
|
25
|
+
properties: {
|
|
26
|
+
query: {
|
|
27
|
+
type: "string",
|
|
28
|
+
pattern: "^([a-zA-Z0-9]([a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?\\.)+[a-zA-Z]{2,}$",
|
|
29
|
+
minLength: 4,
|
|
30
|
+
maxLength: 253,
|
|
31
|
+
description: "Domain or subdomain to analyze"
|
|
32
|
+
},
|
|
33
|
+
searchType: {
|
|
34
|
+
type: "string",
|
|
35
|
+
enum: ["domain", "domain_with_subdomains"],
|
|
36
|
+
default: "domain",
|
|
37
|
+
description: "Type of search query"
|
|
38
|
+
}
|
|
39
|
+
},
|
|
40
|
+
required: ["query"],
|
|
41
|
+
additionalProperties: false
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
async handle(call) {
|
|
45
|
+
try {
|
|
46
|
+
const params = validation_js_1.backlinksSummarySchema.parse(call.arguments);
|
|
47
|
+
const result = await this.backlinksService.getBacklinksSummary(params);
|
|
48
|
+
return this.createSuccessResponse(result);
|
|
49
|
+
}
|
|
50
|
+
catch (error) {
|
|
51
|
+
if (error instanceof zod_1.z.ZodError) {
|
|
52
|
+
return this.createErrorResponse(new Error(`Invalid parameters: ${error.errors.map(e => `${e.path.join('.')}: ${e.message}`).join(', ')}`));
|
|
53
|
+
}
|
|
54
|
+
return this.createErrorResponse(error);
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
exports.BacklinksSummaryHandler = BacklinksSummaryHandler;
|
|
59
|
+
//# sourceMappingURL=backlinks_tools.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"backlinks_tools.js","sourceRoot":"","sources":["../../src/handlers/backlinks_tools.ts"],"names":[],"mappings":";;;AAAA,uCAAwC;AACxC,uEAAkE;AAElE,0DAAwF;AACxF,kDAAgD;AAChD,6BAAwB;AAExB,MAAa,uBAAwB,SAAQ,qBAAW;IAC5C,gBAAgB,CAAmB;IAE3C;QACI,KAAK,EAAE,CAAC;QACR,MAAM,MAAM,GAAG,IAAA,sBAAU,GAAE,CAAC;QAC5B,IAAI,CAAC,gBAAgB,GAAG,IAAI,qCAAgB,CAAC,MAAM,CAAC,CAAC;IACzD,CAAC;IAED,OAAO;QACH,OAAO,uBAAuB,CAAC;IACnC,CAAC;IAED,cAAc;QACV,OAAO,6KAA6K,CAAC;IACzL,CAAC;IAED,cAAc;QACV,OAAO;YACH,IAAI,EAAE,QAAQ;YACd,UAAU,EAAE;gBACR,KAAK,EAAE;oBACH,IAAI,EAAE,QAAQ;oBACd,OAAO,EAAE,iEAAiE;oBAC1E,SAAS,EAAE,CAAC;oBACZ,SAAS,EAAE,GAAG;oBACd,WAAW,EAAE,gCAAgC;iBAChD;gBACD,UAAU,EAAE;oBACR,IAAI,EAAE,QAAQ;oBACd,IAAI,EAAE,CAAC,QAAQ,EAAE,wBAAwB,CAAC;oBAC1C,OAAO,EAAE,QAAQ;oBACjB,WAAW,EAAE,sBAAsB;iBACtC;aACJ;YACD,QAAQ,EAAE,CAAC,OAAO,CAAC;YACnB,oBAAoB,EAAE,KAAK;SAC9B,CAAC;IACN,CAAC;IAED,KAAK,CAAC,MAAM,CAAC,IAAiB;QAC1B,IAAI,CAAC;YACD,MAAM,MAAM,GAAG,sCAAsB,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAA2B,CAAC;YACtF,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,gBAAgB,CAAC,mBAAmB,CAAC,MAAM,CAAC,CAAC;YACvE,OAAO,IAAI,CAAC,qBAAqB,CAAC,MAAM,CAAC,CAAC;QAC9C,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACb,IAAI,KAAK,YAAY,OAAC,CAAC,QAAQ,EAAE,CAAC;gBAC9B,OAAO,IAAI,CAAC,mBAAmB,CAAC,IAAI,KAAK,CAAC,uBAAuB,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC;YAC/I,CAAC;YACD,OAAO,IAAI,CAAC,mBAAmB,CAAC,KAAc,CAAC,CAAC;QACpD,CAAC;IACL,CAAC;CACJ;AApDD,0DAoDC"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { MCPToolCall, MCPToolResponse } from '../types/mcp.js';
|
|
2
|
+
export declare abstract class BaseHandler {
|
|
3
|
+
abstract getName(): string;
|
|
4
|
+
abstract getDescription(): string;
|
|
5
|
+
abstract getInputSchema(): object;
|
|
6
|
+
abstract handle(call: MCPToolCall): Promise<MCPToolResponse>;
|
|
7
|
+
protected createSuccessResponse(data: any): MCPToolResponse;
|
|
8
|
+
protected createErrorResponse(error: Error): MCPToolResponse;
|
|
9
|
+
}
|
|
10
|
+
//# sourceMappingURL=base.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../src/handlers/base.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAG/D,8BAAsB,WAAW;IAC7B,QAAQ,CAAC,OAAO,IAAI,MAAM;IAC1B,QAAQ,CAAC,cAAc,IAAI,MAAM;IACjC,QAAQ,CAAC,cAAc,IAAI,MAAM;IACjC,QAAQ,CAAC,MAAM,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,eAAe,CAAC;IAE5D,SAAS,CAAC,qBAAqB,CAAC,IAAI,EAAE,GAAG,GAAG,eAAe;IAS3D,SAAS,CAAC,mBAAmB,CAAC,KAAK,EAAE,KAAK,GAAG,eAAe;CAU/D"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.BaseHandler = void 0;
|
|
4
|
+
const logger_js_1 = require("../utils/logger.js");
|
|
5
|
+
class BaseHandler {
|
|
6
|
+
createSuccessResponse(data) {
|
|
7
|
+
return {
|
|
8
|
+
content: [{
|
|
9
|
+
type: "text",
|
|
10
|
+
text: JSON.stringify(data, null, 2)
|
|
11
|
+
}]
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
createErrorResponse(error) {
|
|
15
|
+
logger_js_1.logger.error(`Tool ${this.getName()} error:`, error);
|
|
16
|
+
return {
|
|
17
|
+
content: [{
|
|
18
|
+
type: "text",
|
|
19
|
+
text: `Error: ${error.message}`
|
|
20
|
+
}],
|
|
21
|
+
isError: true
|
|
22
|
+
};
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
exports.BaseHandler = BaseHandler;
|
|
26
|
+
//# sourceMappingURL=base.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"base.js","sourceRoot":"","sources":["../../src/handlers/base.ts"],"names":[],"mappings":";;;AACA,kDAA4C;AAE5C,MAAsB,WAAW;IAMnB,qBAAqB,CAAC,IAAS;QACrC,OAAO;YACH,OAAO,EAAE,CAAC;oBACN,IAAI,EAAE,MAAM;oBACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;iBACtC,CAAC;SACL,CAAC;IACN,CAAC;IAES,mBAAmB,CAAC,KAAY;QACtC,kBAAM,CAAC,KAAK,CAAC,QAAQ,IAAI,CAAC,OAAO,EAAE,SAAS,EAAE,KAAK,CAAC,CAAC;QACrD,OAAO;YACH,OAAO,EAAE,CAAC;oBACN,IAAI,EAAE,MAAM;oBACZ,IAAI,EAAE,UAAU,KAAK,CAAC,OAAO,EAAE;iBAClC,CAAC;YACF,OAAO,EAAE,IAAI;SAChB,CAAC;IACN,CAAC;CACJ;AAzBD,kCAyBC"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { BaseHandler } from './base.js';
|
|
2
|
+
import { MCPToolCall, MCPToolResponse } from '../types/mcp.js';
|
|
3
|
+
export declare class DomainsInfoHandler extends BaseHandler {
|
|
4
|
+
private domainService;
|
|
5
|
+
constructor();
|
|
6
|
+
getName(): string;
|
|
7
|
+
getDescription(): string;
|
|
8
|
+
getInputSchema(): object;
|
|
9
|
+
handle(call: MCPToolCall): Promise<MCPToolResponse>;
|
|
10
|
+
}
|
|
11
|
+
//# sourceMappingURL=domain.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"domain.d.ts","sourceRoot":"","sources":["../../src/handlers/domain.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AAExC,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAK/D,qBAAa,kBAAmB,SAAQ,WAAW;IAC/C,OAAO,CAAC,aAAa,CAAgB;;IAQrC,OAAO,IAAI,MAAM;IAIjB,cAAc,IAAI,MAAM;IAIxB,cAAc,IAAI,MAAM;IAuClB,MAAM,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,eAAe,CAAC;CAY5D"}
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.DomainsInfoHandler = void 0;
|
|
4
|
+
const base_js_1 = require("./base.js");
|
|
5
|
+
const domain_js_1 = require("../services/domain.js");
|
|
6
|
+
const validation_js_1 = require("../utils/validation.js");
|
|
7
|
+
const config_js_1 = require("../utils/config.js");
|
|
8
|
+
const zod_1 = require("zod");
|
|
9
|
+
class DomainsInfoHandler extends base_js_1.BaseHandler {
|
|
10
|
+
domainService;
|
|
11
|
+
constructor() {
|
|
12
|
+
super();
|
|
13
|
+
const config = (0, config_js_1.loadConfig)();
|
|
14
|
+
this.domainService = new domain_js_1.DomainService(config);
|
|
15
|
+
}
|
|
16
|
+
getName() {
|
|
17
|
+
return 'serpstat_domains_info';
|
|
18
|
+
}
|
|
19
|
+
getDescription() {
|
|
20
|
+
return 'Get comprehensive SEO information for multiple domains including visibility, keywords, traffic, and dynamics';
|
|
21
|
+
}
|
|
22
|
+
getInputSchema() {
|
|
23
|
+
return {
|
|
24
|
+
type: "object",
|
|
25
|
+
properties: {
|
|
26
|
+
domains: {
|
|
27
|
+
type: "array",
|
|
28
|
+
items: { type: "string" },
|
|
29
|
+
description: "List of domains to analyze (1-10 domains)",
|
|
30
|
+
minItems: 1,
|
|
31
|
+
maxItems: 10
|
|
32
|
+
},
|
|
33
|
+
se: {
|
|
34
|
+
type: "string",
|
|
35
|
+
enum: [
|
|
36
|
+
'g_us', 'g_uk', 'g_de', 'g_fr', 'g_es', 'g_it', 'g_ca', 'g_au',
|
|
37
|
+
'g_nl', 'g_be', 'g_dk', 'g_se', 'g_no', 'g_fi', 'g_pl', 'g_cz',
|
|
38
|
+
'g_ua', 'g_ru', 'bing_us'
|
|
39
|
+
],
|
|
40
|
+
description: "Search engine database (e.g., g_us for Google US)"
|
|
41
|
+
},
|
|
42
|
+
filters: {
|
|
43
|
+
type: "object",
|
|
44
|
+
properties: {
|
|
45
|
+
traff: { type: "number", description: "Exact traffic value" },
|
|
46
|
+
traff_from: { type: "number", description: "Minimum traffic value" },
|
|
47
|
+
traff_to: { type: "number", description: "Maximum traffic value" },
|
|
48
|
+
visible: { type: "number", description: "Exact visibility value" },
|
|
49
|
+
visible_from: { type: "number", description: "Minimum visibility value" },
|
|
50
|
+
visible_to: { type: "number", description: "Maximum visibility value" }
|
|
51
|
+
},
|
|
52
|
+
additionalProperties: false,
|
|
53
|
+
description: "Optional filters for the results"
|
|
54
|
+
}
|
|
55
|
+
},
|
|
56
|
+
required: ["domains", "se"],
|
|
57
|
+
additionalProperties: false
|
|
58
|
+
};
|
|
59
|
+
}
|
|
60
|
+
async handle(call) {
|
|
61
|
+
try {
|
|
62
|
+
const params = validation_js_1.domainsInfoSchema.parse(call.arguments);
|
|
63
|
+
const result = await this.domainService.getDomainsInfo(params);
|
|
64
|
+
return this.createSuccessResponse(result);
|
|
65
|
+
}
|
|
66
|
+
catch (error) {
|
|
67
|
+
if (error instanceof zod_1.z.ZodError) {
|
|
68
|
+
return this.createErrorResponse(new Error(`Invalid parameters: ${error.errors.map(e => `${e.path.join('.')}: ${e.message}`).join(', ')}`));
|
|
69
|
+
}
|
|
70
|
+
return this.createErrorResponse(error);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
exports.DomainsInfoHandler = DomainsInfoHandler;
|
|
75
|
+
//# sourceMappingURL=domain.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"domain.js","sourceRoot":"","sources":["../../src/handlers/domain.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { BaseHandler } from './base.js';
|
|
2
|
+
import { MCPToolCall, MCPToolResponse } from '../types/mcp.js';
|
|
3
|
+
export declare class DomainsInfoHandler extends BaseHandler {
|
|
4
|
+
private domainService;
|
|
5
|
+
constructor();
|
|
6
|
+
getName(): string;
|
|
7
|
+
getDescription(): string;
|
|
8
|
+
getInputSchema(): object;
|
|
9
|
+
handle(call: MCPToolCall): Promise<MCPToolResponse>;
|
|
10
|
+
}
|
|
11
|
+
export declare class CompetitorsHandler extends BaseHandler {
|
|
12
|
+
private domainService;
|
|
13
|
+
constructor();
|
|
14
|
+
getName(): string;
|
|
15
|
+
getDescription(): string;
|
|
16
|
+
getInputSchema(): object;
|
|
17
|
+
handle(call: MCPToolCall): Promise<MCPToolResponse>;
|
|
18
|
+
}
|
|
19
|
+
export declare class DomainKeywordsHandler extends BaseHandler {
|
|
20
|
+
private domainService;
|
|
21
|
+
constructor();
|
|
22
|
+
getName(): string;
|
|
23
|
+
getDescription(): string;
|
|
24
|
+
getInputSchema(): object;
|
|
25
|
+
handle(call: MCPToolCall): Promise<MCPToolResponse>;
|
|
26
|
+
}
|
|
27
|
+
export declare class DomainUrlsHandler extends BaseHandler {
|
|
28
|
+
private domainService;
|
|
29
|
+
constructor();
|
|
30
|
+
getName(): string;
|
|
31
|
+
getDescription(): string;
|
|
32
|
+
getInputSchema(): object;
|
|
33
|
+
handle(call: MCPToolCall): Promise<MCPToolResponse>;
|
|
34
|
+
}
|
|
35
|
+
export declare class DomainRegionsCountHandler extends BaseHandler {
|
|
36
|
+
private domainService;
|
|
37
|
+
constructor();
|
|
38
|
+
getName(): string;
|
|
39
|
+
getDescription(): string;
|
|
40
|
+
getInputSchema(): object;
|
|
41
|
+
handle(call: MCPToolCall): Promise<MCPToolResponse>;
|
|
42
|
+
}
|
|
43
|
+
export declare class GetDomainUniqKeywordsHandler extends BaseHandler {
|
|
44
|
+
private domainService;
|
|
45
|
+
constructor();
|
|
46
|
+
getName(): string;
|
|
47
|
+
getDescription(): string;
|
|
48
|
+
getInputSchema(): object;
|
|
49
|
+
handle(call: MCPToolCall): Promise<MCPToolResponse>;
|
|
50
|
+
}
|
|
51
|
+
//# sourceMappingURL=domain_tools.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"domain_tools.d.ts","sourceRoot":"","sources":["../../src/handlers/domain_tools.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AAExC,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAK/D,qBAAa,kBAAmB,SAAQ,WAAW;IAC/C,OAAO,CAAC,aAAa,CAAgB;;IAQrC,OAAO,IAAI,MAAM;IAIjB,cAAc,IAAI,MAAM;IAIxB,cAAc,IAAI,MAAM;IA4ClB,MAAM,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,eAAe,CAAC;CAY5D;AAED,qBAAa,kBAAmB,SAAQ,WAAW;IAC/C,OAAO,CAAC,aAAa,CAAgB;;IAQrC,OAAO,IAAI,MAAM;IAIjB,cAAc,IAAI,MAAM;IAIxB,cAAc,IAAI,MAAM;IAqDlB,MAAM,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,eAAe,CAAC;CAY5D;AAED,qBAAa,qBAAsB,SAAQ,WAAW;IAClD,OAAO,CAAC,aAAa,CAAgB;;IAQrC,OAAO,IAAI,MAAM;IAIjB,cAAc,IAAI,MAAM;IAIxB,cAAc,IAAI,MAAM;IA6FlB,MAAM,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,eAAe,CAAC;CAY5D;AAED,qBAAa,iBAAkB,SAAQ,WAAW;IAC9C,OAAO,CAAC,aAAa,CAAgB;;IAQrC,OAAO,IAAI,MAAM;IAIjB,cAAc,IAAI,MAAM;IAIxB,cAAc,IAAI,MAAM;IA+ClB,MAAM,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,eAAe,CAAC;CAY5D;AAED,qBAAa,yBAA0B,SAAQ,WAAW;IACtD,OAAO,CAAC,aAAa,CAAgB;;IAQrC,OAAO,IAAI,MAAM;IAIjB,cAAc,IAAI,MAAM;IAMxB,cAAc,IAAI,MAAM;IA6BlB,MAAM,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,eAAe,CAAC;CAY5D;AAED,qBAAa,4BAA6B,SAAQ,WAAW;IACzD,OAAO,CAAC,aAAa,CAAgB;;IAQrC,OAAO,IAAI,MAAM;IAIjB,cAAc,IAAI,MAAM;IAIxB,cAAc,IAAI,MAAM;IAmGlB,MAAM,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,eAAe,CAAC;CAY5D"}
|