feat: wire up swagger-jsdoc dynamic spec, delete static openapi.json
Some checks failed
Build & Deploy to Staging / Build & Deploy to Staging (push) Has been cancelled
Some checks failed
Build & Deploy to Staging / Build & Deploy to Staging (push) Has been cancelled
- Create src/swagger.ts config module for swagger-jsdoc
- Add GET /openapi.json dynamic route (generated from @openapi annotations)
- Delete static public/openapi.json (was drifting from code)
- Add @openapi annotation for deprecated /v1/signup/free in index.ts
- Import swaggerSpec into index.ts
- All 12 endpoints now code-driven: demo/html, demo/markdown, convert/html,
convert/markdown, convert/url, templates, templates/{id}/render,
recover, recover/verify, billing/checkout, signup/free, health
This commit is contained in:
parent
792e2d9142
commit
825c6562ba
11 changed files with 624 additions and 1070 deletions
153
dist/routes/convert.js
vendored
153
dist/routes/convert.js
vendored
|
|
@ -41,7 +41,55 @@ function sanitizeFilename(name) {
|
|||
return name.replace(/[\x00-\x1f"\\\r\n]/g, "").trim() || "document.pdf";
|
||||
}
|
||||
export const convertRouter = Router();
|
||||
// POST /v1/convert/html
|
||||
/**
|
||||
* @openapi
|
||||
* /v1/convert/html:
|
||||
* post:
|
||||
* tags: [Conversion]
|
||||
* summary: Convert HTML to PDF
|
||||
* description: Converts HTML content to a PDF document. Bare HTML fragments are automatically wrapped in a full HTML document.
|
||||
* security:
|
||||
* - BearerAuth: []
|
||||
* - ApiKeyHeader: []
|
||||
* requestBody:
|
||||
* required: true
|
||||
* content:
|
||||
* application/json:
|
||||
* schema:
|
||||
* allOf:
|
||||
* - type: object
|
||||
* required: [html]
|
||||
* properties:
|
||||
* html:
|
||||
* type: string
|
||||
* description: HTML content to convert. Can be a full document or a fragment.
|
||||
* example: '<h1>Hello World</h1><p>My first PDF</p>'
|
||||
* css:
|
||||
* type: string
|
||||
* description: Optional CSS to inject (only used when html is a fragment, not a full document)
|
||||
* example: 'body { font-family: sans-serif; padding: 40px; }'
|
||||
* - $ref: '#/components/schemas/PdfOptions'
|
||||
* responses:
|
||||
* 200:
|
||||
* description: PDF document
|
||||
* content:
|
||||
* application/pdf:
|
||||
* schema:
|
||||
* type: string
|
||||
* format: binary
|
||||
* 400:
|
||||
* description: Missing html field
|
||||
* 401:
|
||||
* description: Missing API key
|
||||
* 403:
|
||||
* description: Invalid API key
|
||||
* 415:
|
||||
* description: Unsupported Content-Type (must be application/json)
|
||||
* 429:
|
||||
* description: Rate limit or usage limit exceeded
|
||||
* 500:
|
||||
* description: PDF generation failed
|
||||
*/
|
||||
convertRouter.post("/html", async (req, res) => {
|
||||
let slotAcquired = false;
|
||||
try {
|
||||
|
|
@ -90,7 +138,54 @@ convertRouter.post("/html", async (req, res) => {
|
|||
}
|
||||
}
|
||||
});
|
||||
// POST /v1/convert/markdown
|
||||
/**
|
||||
* @openapi
|
||||
* /v1/convert/markdown:
|
||||
* post:
|
||||
* tags: [Conversion]
|
||||
* summary: Convert Markdown to PDF
|
||||
* description: Converts Markdown content to HTML and then to a PDF document.
|
||||
* security:
|
||||
* - BearerAuth: []
|
||||
* - ApiKeyHeader: []
|
||||
* requestBody:
|
||||
* required: true
|
||||
* content:
|
||||
* application/json:
|
||||
* schema:
|
||||
* allOf:
|
||||
* - type: object
|
||||
* required: [markdown]
|
||||
* properties:
|
||||
* markdown:
|
||||
* type: string
|
||||
* description: Markdown content to convert
|
||||
* example: '# Hello World\n\nThis is **bold** and *italic*.'
|
||||
* css:
|
||||
* type: string
|
||||
* description: Optional CSS to inject into the rendered HTML
|
||||
* - $ref: '#/components/schemas/PdfOptions'
|
||||
* responses:
|
||||
* 200:
|
||||
* description: PDF document
|
||||
* content:
|
||||
* application/pdf:
|
||||
* schema:
|
||||
* type: string
|
||||
* format: binary
|
||||
* 400:
|
||||
* description: Missing markdown field
|
||||
* 401:
|
||||
* description: Missing API key
|
||||
* 403:
|
||||
* description: Invalid API key
|
||||
* 415:
|
||||
* description: Unsupported Content-Type
|
||||
* 429:
|
||||
* description: Rate limit or usage limit exceeded
|
||||
* 500:
|
||||
* description: PDF generation failed
|
||||
*/
|
||||
convertRouter.post("/markdown", async (req, res) => {
|
||||
let slotAcquired = false;
|
||||
try {
|
||||
|
|
@ -136,7 +231,59 @@ convertRouter.post("/markdown", async (req, res) => {
|
|||
}
|
||||
}
|
||||
});
|
||||
// POST /v1/convert/url
|
||||
/**
|
||||
* @openapi
|
||||
* /v1/convert/url:
|
||||
* post:
|
||||
* tags: [Conversion]
|
||||
* summary: Convert URL to PDF
|
||||
* description: |
|
||||
* Fetches a URL and converts the rendered page to PDF. JavaScript is disabled for security.
|
||||
* Private/internal IP addresses are blocked (SSRF protection). DNS is pinned to prevent rebinding.
|
||||
* security:
|
||||
* - BearerAuth: []
|
||||
* - ApiKeyHeader: []
|
||||
* requestBody:
|
||||
* required: true
|
||||
* content:
|
||||
* application/json:
|
||||
* schema:
|
||||
* allOf:
|
||||
* - type: object
|
||||
* required: [url]
|
||||
* properties:
|
||||
* url:
|
||||
* type: string
|
||||
* format: uri
|
||||
* description: URL to convert (http or https only)
|
||||
* example: 'https://example.com'
|
||||
* waitUntil:
|
||||
* type: string
|
||||
* enum: [load, domcontentloaded, networkidle0, networkidle2]
|
||||
* default: domcontentloaded
|
||||
* description: When to consider navigation finished
|
||||
* - $ref: '#/components/schemas/PdfOptions'
|
||||
* responses:
|
||||
* 200:
|
||||
* description: PDF document
|
||||
* content:
|
||||
* application/pdf:
|
||||
* schema:
|
||||
* type: string
|
||||
* format: binary
|
||||
* 400:
|
||||
* description: Missing/invalid URL or URL resolves to private IP
|
||||
* 401:
|
||||
* description: Missing API key
|
||||
* 403:
|
||||
* description: Invalid API key
|
||||
* 415:
|
||||
* description: Unsupported Content-Type
|
||||
* 429:
|
||||
* description: Rate limit or usage limit exceeded
|
||||
* 500:
|
||||
* description: PDF generation failed
|
||||
*/
|
||||
convertRouter.post("/url", async (req, res) => {
|
||||
let slotAcquired = false;
|
||||
try {
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue