mirror of
https://github.com/Dokploy/dokploy.git
synced 2026-06-15 20:25:23 +02:00
feat: add OpenAPI generation script and workflow
- Introduced a new script to generate OpenAPI specifications for the Dokploy API. - Added a GitHub Actions workflow to automate the generation and syncing of OpenAPI documentation upon changes in the API routers. - Updated package.json files to include new commands for generating OpenAPI specifications. - Added openapi.json to .gitignore to prevent accidental commits of generated files.
This commit is contained in:
93
.github/workflows/sync-openapi-docs.yml
vendored
Normal file
93
.github/workflows/sync-openapi-docs.yml
vendored
Normal file
@@ -0,0 +1,93 @@
|
|||||||
|
name: Generate and Sync OpenAPI
|
||||||
|
|
||||||
|
on:
|
||||||
|
# Se ejecuta cuando hay cambios en los routers de la API
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- canary
|
||||||
|
paths:
|
||||||
|
- 'apps/dokploy/server/api/routers/**'
|
||||||
|
- 'packages/server/src/services/**'
|
||||||
|
- 'packages/server/src/db/schema/**'
|
||||||
|
|
||||||
|
# Permite ejecución manual
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
generate-and-commit:
|
||||||
|
name: Generate OpenAPI and commit to Dokploy repo
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout Dokploy repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Setup Node.js
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: '20'
|
||||||
|
|
||||||
|
- name: Setup pnpm
|
||||||
|
uses: pnpm/action-setup@v2
|
||||||
|
with:
|
||||||
|
version: 8
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: pnpm install --frozen-lockfile
|
||||||
|
|
||||||
|
- name: Generate OpenAPI specification
|
||||||
|
run: pnpm generate:openapi
|
||||||
|
|
||||||
|
- name: Commit OpenAPI spec
|
||||||
|
run: |
|
||||||
|
git config user.name "Dokploy Bot"
|
||||||
|
git config user.email "bot@dokploy.com"
|
||||||
|
|
||||||
|
# Verifica si el archivo existe
|
||||||
|
if [ ! -f openapi.json ]; then
|
||||||
|
echo "❌ openapi.json not found"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Usa -f para forzar el add de archivos en .gitignore
|
||||||
|
git add -f openapi.json
|
||||||
|
|
||||||
|
# Verifica si hay cambios para commitear
|
||||||
|
if git diff --cached --quiet; then
|
||||||
|
echo "📝 No changes detected in OpenAPI spec"
|
||||||
|
echo "HAS_CHANGES=false" >> $GITHUB_ENV
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "HAS_CHANGES=true" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
# Commit los cambios
|
||||||
|
git commit -m "chore: update OpenAPI specification [skip ci]
|
||||||
|
|
||||||
|
Generated from commit: ${{ github.sha }}
|
||||||
|
Triggered by: ${{ github.event_name }}"
|
||||||
|
|
||||||
|
git push
|
||||||
|
|
||||||
|
echo "✅ OpenAPI spec committed successfully"
|
||||||
|
|
||||||
|
- name: Trigger website sync
|
||||||
|
if: env.HAS_CHANGES == 'true'
|
||||||
|
uses: peter-evans/repository-dispatch@v2
|
||||||
|
with:
|
||||||
|
token: ${{ secrets.DOCS_SYNC_TOKEN }}
|
||||||
|
repository: dokploy/website # Cambia por tu repo de docs
|
||||||
|
event-type: openapi-updated
|
||||||
|
client-payload: '{"commit": "${{ github.sha }}", "timestamp": "${{ github.event.head_commit.timestamp }}"}'
|
||||||
|
|
||||||
|
- name: Create summary
|
||||||
|
run: |
|
||||||
|
echo "## 📊 OpenAPI Generation Summary" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "- **Repository:** \`${{ github.repository }}\`" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "- **Commit:** \`${{ github.sha }}\`" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "- **Trigger:** \`${{ github.event_name }}\`" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "- **Changes:** \`${{ env.HAS_CHANGES }}\`" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "- **Status:** ✅ Success" >> $GITHUB_STEP_SUMMARY
|
||||||
|
|
||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -13,6 +13,8 @@ node_modules
|
|||||||
.env.test.local
|
.env.test.local
|
||||||
.env.production.local
|
.env.production.local
|
||||||
|
|
||||||
|
openapi.json
|
||||||
|
|
||||||
# Testing
|
# Testing
|
||||||
coverage
|
coverage
|
||||||
|
|
||||||
|
|||||||
@@ -34,7 +34,8 @@
|
|||||||
"docker:build:canary": "./docker/build.sh canary",
|
"docker:build:canary": "./docker/build.sh canary",
|
||||||
"docker:push:canary": "./docker/push.sh canary",
|
"docker:push:canary": "./docker/push.sh canary",
|
||||||
"version": "echo $(node -p \"require('./package.json').version\")",
|
"version": "echo $(node -p \"require('./package.json').version\")",
|
||||||
"test": "vitest --config __test__/vitest.config.ts"
|
"test": "vitest --config __test__/vitest.config.ts",
|
||||||
|
"generate:openapi": "tsx -r dotenv/config scripts/generate-openapi.ts"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@ai-sdk/anthropic": "^2.0.5",
|
"@ai-sdk/anthropic": "^2.0.5",
|
||||||
|
|||||||
132
apps/dokploy/scripts/generate-openapi.ts
Normal file
132
apps/dokploy/scripts/generate-openapi.ts
Normal file
@@ -0,0 +1,132 @@
|
|||||||
|
#!/usr/bin/env tsx
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Script to generate OpenAPI specification locally
|
||||||
|
* This runs in CI/CD to generate the openapi.json file
|
||||||
|
* which can then be consumed by the documentation website
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { writeFileSync } from "node:fs";
|
||||||
|
import { dirname, resolve } from "node:path";
|
||||||
|
import { fileURLToPath } from "node:url";
|
||||||
|
import { generateOpenApiDocument } from "@dokploy/trpc-openapi";
|
||||||
|
import { appRouter } from "../server/api/root";
|
||||||
|
|
||||||
|
const __filename = fileURLToPath(import.meta.url);
|
||||||
|
const __dirname = dirname(__filename);
|
||||||
|
|
||||||
|
async function generateOpenAPI() {
|
||||||
|
try {
|
||||||
|
console.log("🔄 Generating OpenAPI specification...");
|
||||||
|
|
||||||
|
const openApiDocument = generateOpenApiDocument(appRouter, {
|
||||||
|
title: "Dokploy API",
|
||||||
|
version: "1.0.0",
|
||||||
|
baseUrl: "https://your-dokploy-instance.com/api",
|
||||||
|
docsUrl: "https://docs.dokploy.com/api",
|
||||||
|
tags: [
|
||||||
|
"admin",
|
||||||
|
"docker",
|
||||||
|
"compose",
|
||||||
|
"registry",
|
||||||
|
"cluster",
|
||||||
|
"user",
|
||||||
|
"domain",
|
||||||
|
"destination",
|
||||||
|
"backup",
|
||||||
|
"deployment",
|
||||||
|
"mounts",
|
||||||
|
"certificates",
|
||||||
|
"settings",
|
||||||
|
"security",
|
||||||
|
"redirects",
|
||||||
|
"port",
|
||||||
|
"project",
|
||||||
|
"application",
|
||||||
|
"mysql",
|
||||||
|
"postgres",
|
||||||
|
"redis",
|
||||||
|
"mongo",
|
||||||
|
"mariadb",
|
||||||
|
"sshRouter",
|
||||||
|
"gitProvider",
|
||||||
|
"bitbucket",
|
||||||
|
"github",
|
||||||
|
"gitlab",
|
||||||
|
"gitea",
|
||||||
|
"server",
|
||||||
|
"swarm",
|
||||||
|
"ai",
|
||||||
|
"organization",
|
||||||
|
"schedule",
|
||||||
|
"rollback",
|
||||||
|
"volumeBackups",
|
||||||
|
"environment",
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
// Enhance metadata
|
||||||
|
openApiDocument.info = {
|
||||||
|
title: "Dokploy API",
|
||||||
|
description:
|
||||||
|
"Complete API documentation for Dokploy - Deploy applications, manage databases, and orchestrate your infrastructure. This API allows you to programmatically manage all aspects of your Dokploy instance.",
|
||||||
|
version: "1.0.0",
|
||||||
|
contact: {
|
||||||
|
name: "Dokploy Team",
|
||||||
|
url: "https://dokploy.com",
|
||||||
|
},
|
||||||
|
license: {
|
||||||
|
name: "Apache 2.0",
|
||||||
|
url: "https://github.com/dokploy/dokploy/blob/canary/LICENSE",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
// Add security schemes
|
||||||
|
openApiDocument.components = {
|
||||||
|
...openApiDocument.components,
|
||||||
|
securitySchemes: {
|
||||||
|
apiKey: {
|
||||||
|
type: "apiKey",
|
||||||
|
in: "header",
|
||||||
|
name: "x-api-key",
|
||||||
|
description:
|
||||||
|
"API key authentication. Generate an API key from your Dokploy dashboard under Settings > API Keys.",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
// Apply global security
|
||||||
|
openApiDocument.security = [
|
||||||
|
{
|
||||||
|
apiKey: [],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
// Add external docs
|
||||||
|
openApiDocument.externalDocs = {
|
||||||
|
description: "Full documentation",
|
||||||
|
url: "https://docs.dokploy.com",
|
||||||
|
};
|
||||||
|
|
||||||
|
// Write to root of repo
|
||||||
|
const outputPath = resolve(__dirname, "../../../openapi.json");
|
||||||
|
writeFileSync(
|
||||||
|
outputPath,
|
||||||
|
JSON.stringify(openApiDocument, null, 2),
|
||||||
|
"utf-8",
|
||||||
|
);
|
||||||
|
|
||||||
|
console.log("✅ OpenAPI specification generated successfully!");
|
||||||
|
console.log(`📄 Output: ${outputPath}`);
|
||||||
|
console.log(
|
||||||
|
`📊 Endpoints: ${Object.keys(openApiDocument.paths || {}).length}`,
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("❌ Error generating OpenAPI specification:", error);
|
||||||
|
process.exit(1);
|
||||||
|
} finally {
|
||||||
|
process.exit(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
generateOpenAPI();
|
||||||
@@ -20,7 +20,8 @@
|
|||||||
"build": "pnpm -r run build",
|
"build": "pnpm -r run build",
|
||||||
"format-and-lint": "biome check .",
|
"format-and-lint": "biome check .",
|
||||||
"check": "biome check --write --no-errors-on-unmatched --files-ignore-unknown=true",
|
"check": "biome check --write --no-errors-on-unmatched --files-ignore-unknown=true",
|
||||||
"format-and-lint:fix": "biome check . --write"
|
"format-and-lint:fix": "biome check . --write",
|
||||||
|
"generate:openapi": "pnpm --filter=dokploy run generate:openapi"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@biomejs/biome": "2.1.1",
|
"@biomejs/biome": "2.1.1",
|
||||||
|
|||||||
Reference in New Issue
Block a user