CLAUDE.md•12.1 kB
# CLAUDE.md
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
## Project Overview
This is a TypeScript/Node.js-based MCP (Model Context Protocol) server for OpenSubtitles API integration. The server provides subtitle search and download functionality with a freemium model, using the existing Kong gateway at api.opensubtitles.com for API management.
## Commands
### Development Commands
```bash
npm install                # Install dependencies
npm run build              # Build TypeScript to JavaScript
npm run watch              # Development with auto-rebuild
npm run dev                # Build and run the server
npm test                   # Run tests (currently placeholder)
npm run test:evals         # Run MCP evaluation tests (when implemented)
npm run inspector          # Debug with MCP Inspector
npm run prepublish         # Build before publishing
```
### Testing and Debugging
```bash
npx @modelcontextprotocol/inspector dist/index.js  # MCP Inspector for debugging
npx mcp-eval src/evals/evals.ts dist/index.js       # Run evaluation tests
```
## Architecture
### Core Structure (Planned)
- **MCP Server**: TypeScript implementation using @modelcontextprotocol/sdk
- **API Integration**: All requests go through Kong gateway (api.opensubtitles.com)
- **Rate Limiting**: Handled by Kong configuration
- **Deployment**: mcp.opensubtitles.com domain
- **Distribution**: NPM package @opensubtitles/mcp-server
### Planned Directory Structure
```
src/
├── index.ts               # Main MCP server entry point
├── server.ts              # MCP server implementation
├── api-client.ts          # Kong gateway client wrapper
├── tools/                 # MCP tool implementations
│   ├── search-subtitles.ts
│   ├── download-subtitle.ts
│   └── calculate-file-hash.ts
└── utils/                 # Helper functions
    ├── hash-calculator.ts
    └── subtitle-parser.ts
```
### MCP Tools to Implement
1. **search_subtitles**: Search with all OpenSubtitles API parameters (query, imdb_id, tmdb_id, moviehash, etc.)
2. **download_subtitle**: Download subtitle content (rate limited via Kong)
3. **calculate_file_hash**: Calculate OpenSubtitles hash for movie files
## Technical Details
### Dependencies
- @modelcontextprotocol/sdk@1.17.5 - MCP TypeScript SDK
- axios@1.11.0 - HTTP client for Kong gateway communication
- zod@3.25.76 - Schema validation and type safety
- dotenv@16.6.1 - Environment configuration
- express@4.21.2 - HTTP server for development and debugging
- cors@2.8.5 - CORS middleware for HTTP mode
- crypto - File hashing (built-in Node.js)
### API Integration
- Base URL: https://api.opensubtitles.com (Kong gateway)
- VIP URL: https://vip-api.opensubtitles.com/api/v1 (for authenticated users)
- Rate limiting: Managed by Kong (unlimited search, limited downloads)
- Authentication: User API keys passed through Kong
- **Official API Specification**: Use `curl https://stoplight.io/api/v1/projects/opensubtitles/opensubtitles-api/nodes/open_api.json` for complete OpenAPI 3.0.3 specification
### Business Model
- Anonymous users: Unlimited search, 0 downloads
- API key users: Full OpenSubtitles API access
- Future premium: Server-side API key with higher limits
## Configuration
### Environment Variables
```
OPENSUBTITLES_API_BASE=https://api.opensubtitles.com
NODE_ENV=production
PORT=1620
```
### TypeScript Configuration
- Version: 5.9.2
- Target: ES2022
- Module: ESNext
- Strict mode enabled
- Output directory: dist/
## Installation and Usage
### Docker Development Setup (Recommended)
#### Prerequisites
- Docker and Docker Compose installed
- n8n and Qdrant containers (optional - included in compose)
#### Quick Start
```bash
# Clone repository
git clone https://github.com/opensubtitles/mcp.opensubtitles.com.git
cd mcp.opensubtitles.com
# Start all services
docker-compose up -d
# Or start only OpenSubtitles MCP server
docker-compose up opensubtitles
# View logs
docker-compose logs -f opensubtitles
```
#### Docker Services
- **opensubtitles**: MCP server on `http://opensubtitles:1620` ✅
- **n8n**: Workflow automation on `http://localhost:5678` (optional)
- **qdrant**: Vector database on `http://localhost:6333` (optional)
- **postgres**: Database for n8n (optional)
#### Development Workflow
1. Edit source files locally in `./src/`
2. TypeScript auto-compiles with watch mode ✅
3. Server automatically restarts on changes ✅
4. Test changes immediately in containers ✅
5. Volume mapping preserves node_modules performance ✅
#### Tested Endpoints
- **Health Check**: `http://localhost:1620/health` ✅
- **Tools List**: `http://localhost:1620/tools` ✅
- **HTTP Proxy**: `POST http://localhost:1620/proxy` ✅
- **Web Interface**: `http://localhost:1620/web` ✅
- **SSE (for MCP)**: `http://localhost:1620/sse` ✅
- **Streamable (for MCP)**: `http://localhost:1620/message` ✅
#### n8n Integration
**MCP Client Configuration (SSE - Legacy):**
- **Endpoint**: `http://opensubtitles:1620/sse`
- **Transport**: Server Sent Events
- **Network**: `mcp_network` (automatic)
**MCP Client Configuration (Streamable - Modern):**
- **Endpoint**: `http://opensubtitles:1620/message`
- **Transport**: HTTP Streamable
- **Network**: `mcp_network` (automatic)
**HTTP Tool Configuration:**
- **URL**: `http://opensubtitles:1620/proxy`
- **Method**: POST
- **Content-Type**: application/json
- **Body Example**:
```json
{
  "tool": "search_subtitles",
  "arguments": {
    "query": "Matrix",
    "year": 1999,
    "languages": "en,es"
  }
}
```
#### Container Network Access
All containers communicate via hostnames:
- `opensubtitles` → OpenSubtitles MCP server
- `n8n` → n8n workflow automation
- `qdrant` → Vector database
- `postgres` → PostgreSQL database
### NPM Package Usage
```bash
npx @opensubtitles/mcp-server                    # Direct usage
npx @michaellatman/mcp-get@latest install @opensubtitles/mcp-server  # Via mcp-get
```
### Claude Desktop Integration
```json
{
  "mcpServers": {
    "opensubtitles": {
      "command": "npx",
      "args": ["-y", "@opensubtitles/mcp-server"]
    }
  }
}
```
## Development Notes
- Complete TypeScript MCP server implementation with 3 tools
- Kong gateway handles all rate limiting and API management
- Use MCP Inspector for debugging (stdio communication)
- Follow security best practices for API key handling
- Supports both HTTP mode (port 1620) and stdio mode for Claude Desktop
### API Reference and Specifications
When working with the OpenSubtitles API, always refer to the official specification:
```bash
# Get complete OpenAPI 3.0.3 specification
curl https://stoplight.io/api/v1/projects/opensubtitles/opensubtitles-api/nodes/open_api.json
```
This specification includes:
- **Complete endpoint documentation** for all `/api/v1/*` endpoints
- **Request/response schemas** with exact parameter names and types
- **Authentication requirements** for both API keys and JWT tokens
- **Rate limiting information** and quota management
- **Error codes and responses** for proper error handling
- **Example requests/responses** for all endpoints
### Key API Endpoints Implemented
1. **POST /download**: Download subtitle files by `file_id`
   - Requires: `file_id` (from search results)
   - Optional: `sub_format`, `file_name`, `in_fps`, `out_fps`, `timeshift`, `force_download`
   - Authentication: Both `Api-Key` and `Authorization` headers required
2. **GET /subtitles**: Search for subtitles
   - Supports: `query`, `imdb_id`, `tmdb_id`, `parent_imdb_id`, `season_number`, `episode_number`, etc.
   - Rate limit: Unlimited for search operations
3. **POST /login**: Authenticate user to get JWT token
   - Returns: `base_url` (may redirect to VIP server), `token`, user info
## Publishing to NPM
To publish this package to NPM registry so users can use `npx @opensubtitles/mcp-server`:
### Prerequisites
1. Ensure you have NPM account with access to @opensubtitles organization
2. Login to NPM: `npm login`
3. Verify build works: `npm run build`
### Publishing Steps
```bash
# Ensure clean build
npm run build
# Test package locally
npm pack
tar -tf opensubtitles-mcp-server-1.0.0.tgz
# Publish to NPM
npm publish
# Verify publication
npm view @opensubtitles/mcp-server
```
### After Publishing
Users can then use:
```bash
npx @opensubtitles/mcp-server
```
And Claude Desktop config becomes:
```json
{
  "mcpServers": {
    "opensubtitles": {
      "command": "npx",
      "args": ["-y", "@opensubtitles/mcp-server"],
      "env": {
        "MCP_MODE": "stdio"
      }
    }
  }
}
```
### Current Status
- ✅ Full TypeScript implementation complete (v1.4.4)
- ✅ HTTP server mode working on port 1620 with CORS support
- ✅ stdio mode for Claude Desktop integration
- ✅ All three tools implemented (search, download, hash)
- ✅ Comprehensive error handling and logging
- ✅ ES module entry point fix applied (resolves Claude Desktop crashes)
- ✅ Keep-alive mechanisms for stable MCP connection
- ✅ Tested and working with Claude Desktop MCP integration
- ✅ Express/CORS dependencies added for HTTP mode
- ✅ Updated TypeScript to v5.9.2
- ✅ **Docker development environment fully deployed**
- ✅ **Volume mapping for live code editing**
- ✅ **Docker network connectivity with n8n/Qdrant**
- ✅ **All HTTP endpoints tested and working (health, tools, proxy, sse, streamable)**
- ✅ **Streamable HTTP transport fully implemented based on WooCommerce patterns**
- ✅ **Direct chunked transfer encoding with proper JSON-RPC 2.0 compliance**
- ✅ **n8n compatibility confirmed - streamable endpoint working perfectly**
- ✅ **Manual body parsing to avoid Express.js middleware interference**
- ✅ **Both tools/list and tools/call methods working via streamable transport**
- ✅ **Development workflow optimized with auto-rebuild**
- ✅ Package ready for publication
- 🔄 NPM publishing pending (npmjs.com temporary issues)
- 📋 GitHub installation temporary workaround: `npx https://github.com/opensubtitles/mcp.opensubtitles.com.git`
- 🌐 HTTP server available at: https://mcp.opensubtitles.com
### Troubleshooting
#### n8n Integration Issues
If n8n reports "Could not connect to your MCP server":
1. **Use simplified endpoint**: Use `http://localhost:1620/mcp` instead of `http://localhost:1620/message`
2. **Verify User-Agent**: The server automatically detects n8n clients and provides standard JSON responses
3. **Check connection**: Test with curl:
   ```bash
   curl -X POST http://localhost:1620/mcp \
     -H "Content-Type: application/json" \
     -H "User-Agent: n8n" \
     -d '{"jsonrpc":"2.0","method":"tools/list","params":{},"id":1}'
   ```
#### n8n Configuration
For n8n HTTP Streamable Authentication node:
- **URL**: `http://localhost:1620/mcp` (use simplified endpoint)
- **Method**: POST
- **Authentication**: None
- **Headers**: Content-Type: application/json
#### User-Agent Compliance
The server uses `MCPServer v{VERSION}` as User-Agent header to comply with OpenSubtitles API requirements as specified in [OpenSubtitles Best Practices](https://opensubtitles.stoplight.io/docs/opensubtitles-api/6ef2e232095c7-best-practices).
#### Claude Desktop Integration Issues
If the MCP server crashes or doesn't connect properly with Claude Desktop:
1. **Check logs**: Look for MCP server logs in Claude Desktop's MCP log files
2. **ES Module Issues**: The fix has been applied - server now always calls main() instead of relying on ES module entry point detection
3. **Process Exit**: Keep-alive mechanisms are implemented to prevent premature process exit
4. **Debugging**: Use the simple test server at `dist/simple-index.js` for debugging basic MCP connectivity
#### GitHub Installation
Until NPM package is available:
```json
{
  "mcpServers": {
    "opensubtitles": {
      "command": "npx",
      "args": ["-y", "https://github.com/opensubtitles/mcp.opensubtitles.com.git"],
      "env": {
        "MCP_MODE": "stdio"
      }
    }
  }
}
```