Git Product home page Git Product logo

grzechuj / swagger-autogen Goto Github PK

View Code? Open in Web Editor NEW

This project forked from swagger-autogen/swagger-autogen

0.0 1.0 0.0 139 KB

This module performs the automatic construction of the Swagger documentation. The module is able to identify the endpoints and automatically capture methods such as get, post, put, and so on. The module can also identify the paths, routes, middlewares, response status code, parameters in the path, query and body. It is possible to add information such as: endpoint description, parameter description, defininitions, security, among others. It is also possible to ignore or disable the automatic capture of an endpoint (in the latter case, having to manually add each information). The module generates the .json file with the documentation in the swagger format.

License: MIT License

JavaScript 100.00%

swagger-autogen's Introduction

swagger-autogen

This module performs the automatic construction of the Swagger documentation. The module is able to identify the endpoints and automatically capture methods such as get, post, put, and so on. The module can also identify the paths, routes, middlewares, response status code, parameters in the path, query and body. It is possible to add information such as: endpoint description, parameter description, defininitions, security, among others. It is also possible to ignore or disable the automatic capture of an endpoint (in the latter case, having to manually add each information). The module generates the .json file with the documentation in the swagger format.

NPM Version NPM Downloads Known Vulnerabilities

NEWS! (version 2)

Swagger-autogen now recognizes Express Router and TypeScript features.

Contents

Installation

This is a Node.js module available through the npm.

$ npm install --save swagger-autogen

It is loaded using the require() function:

const swaggerAutogen = require('swagger-autogen')()

Update

If you already have the module installed and want to update to the latest version, use the command:

$ npm install --save [email protected]

Usage

Example using Router

Example without Router

See Tutorial in English

See Tutorial em Português Brasil

The two sections below will show the most basic and most complete use of this module.

Function signature:

const swaggerAutogen: (outputFile: <string>, endpointsFiles: <Array of string>, data: <object>) => Promise <any>

outputFile: (Required*). Output file. It will be the file generated by the module containing the documentation in the format identified by Swagger.

endpointsFiles: (Required*). Files containing the endpoints. These are the files that contain methods such as get, post, put, and so on, for example: app.get('/path', ...) or route.post('/path', ...).

doc: (Not Required). Object containing the details of the documentation. If not informed, or if any parameter of the object is omitted, the default values ​​will be used. (See: Usage (With optionals) section)

Usage (Basic)

The code below must be inserted in a separate file, for example: swagger.js. For example:

File: swagger.js

const swaggerAutogen = require('swagger-autogen')()

const doc = {
    info: {
        title: "My API",
        description: "Description"
    },
    host: "localhost:3000",
    schemes: ['http']
}

const outputFile = './path/swagger-output.json'
const endpointsFiles = ['./path/endpointsUser.js', './path/endpointsBook.js']

/* NOTE: if you use the express Router, you must pass in the 
   'endpointsFiles' only the root file where the route starts,
   such as: index.js, app.js, routes.js, ... */

swaggerAutogen(outputFile, endpointsFiles, doc)

Usage (With optionals)

The code below must be inserted in a separate file, for example:

File: swagger.js

const swaggerAutogen = require('swagger-autogen')()

const doc = {
    info: {
        "version": "",                // by default: "1.0.0"
        "title": "",                  // by default: "REST API"
        "description": ""             // by default: ""
    },
    host: "",                         // by default: "localhost:3000"
    basePath: "",                     // by default: "/"
    schemes: [],                      // by default: ['http']
    consumes: [],                     // by default: ['application/json']
    produces: [],                     // by default: ['application/json']
    tags: [                           // by default: empty Array
        {
            "name": "",               // Tag name
            "description": ""         // Tag description
        },
        // { ... }
    ],
    securityDefinitions: { },         // by default: empty object
    definitions: { }                  // by default: empty object
}

const outputFile = './path/swagger-output.json'
const endpointsFiles = ['./path/endpointsUser.js', './path/endpointsBook.js']

/* NOTE: if you use the express Router, you must pass in the 
   'endpointsFiles' only the root file where the route starts,
   such as: index.js, app.js, routes.js, ... */

swaggerAutogen(outputFile, endpointsFiles, doc)

Building documentation without starting the project

To build the documentation without starting your project, add the following script to your project's package.json file:

File: package.json

    // { ... },
    "scripts": {
        // ... ,
        "swagger-autogen": "node ./swagger.js"
    }

Where ./swagger.js is the file containing the swaggerAutogen(...) function call (see section Usage. After that, at the root of your project, run the following command:

$ npm run swagger-autogen

Building documentation at project startup

To build the documentation before the project starts and immediately start it, rewrite the swaggerAutogen(...) function as follows:

// ...
swaggerAutogen(outputFile, endpointsFiles, doc).then( () => {
    require('./index.js')           // Your project's root file
})

Where index.js is your project's root file. Change the start script in your project's package.json to point to the file containing the swaggerAutogen(...) function. If you use Visual Studio Code, change the reference in your launch.json in the same way. Now, just run your project as usual. With that the documentation will be generated and soon after the project will start, making the documentation always updated when the project is started.

See: Complete example

Options

It is possible to change some options of the module by passing an object as a parameter. This object is optional.

const options = {
    language: <string>,         // By default is 'en-US'
    disableLogs: <boolean>,     // By default is false
    disableWarnings: <boolean>  // By default is false
}

const swaggerAutogen = require('swagger-autogen')(options)

To see the available languages, go to the section Response Language

Endpoints

The way to configure the module is done within comments, and can be in the format // ... or /* ... */. The used pattern will be #swagger.something tag. Each comment can contain one or more #swagger.something tags. NOTE: ALL COMMENTS CONTAINING #swagger.something MUST BE WITHIN OF FUNCTIONS.

Automatic capture

In this case it is not necessary to do anything. Considering, for example, if the pattern of your API is as follows:

    ...
    app.post('/users', (req, res) => {
        ...
        users.addUser(req.query.obj)
        ...
        if(...)
            return res.status(201).send(data)
        ...
        return res.status(500).send(false)
    })
    ...

The recognition of the method, path, parameters and status of the response will be automatic.

See an example here!

Tags

To inform which tags the endpoinst belongs to, use the #swagger.tags tag, for example:

    ...
    app.get('/users', (req, res) => {
        ...
        // #swagger.tags = ['Users']
        ...
    })

See an example here!

Summary

This is the summary of the Endpoint. To add it, use the #swagger.summary tag, for example:

    ...
    app.get('/users/:id', (req, res) => {
        ...
        // #swagger.summary = 'Your summary here'
        ...
    })

See an example here!

Description

This is the description of the Endpoint. To add it, use the #swagger.description tag, for example:

    ...
    app.get('/users/:id', (req, res) => {
        ...
        // #swagger.description = 'Endpoint used to obtain a user.'
        ...
    })

See an example here!

Operation ID

This is the operationId of the Endpoint. To add it, use the #swagger.operationId tag, for example:

    ...
    app.get('/users/:id', (req, res) => {
        ...
        // #swagger.operationId = 'Your_operationId_here'
        ...
    })

See an example here!

Consumes and Produces

Use the #swagger.produces = ['contentType'] or #swagger.consumes = ['contentType'] tag to add a new produce or a new consume, respectively. In the Example (Consumes) below, the two endpoints will have the same result in the documentation.

Example (Consumes):

    ...
    app.get('/users/:id', (req, res) => {
        ...
        // Recognizes the 'consumes' automatically
        res.setHeader('Content-Type', 'application/xml')
        ...
    })

OR

    app.get('/users/:id', (req, res) => {
        ...
        // #swagger.consumes = ['application/xml']
        ...
    })

Example (Produces):

    ...
    app.get('/v2/users/:id', (req, res) => {
        ...
        // #swagger.produces = ['application/json']
        ...
    })

See an example here!

Parameters

It is possible to create or complement automatically detected parameters. Use the #swagger.parameters['parameterName'] tag to create a new parameter or to complete an existing parameter (automatically detected).

All optional parameters for the tag parameter:

/* #swagger.parameters['parameterName'] = {
        in: <string>,
        description: <string>,
        required: <boolean>,
        type: <string>,
        format: <string>,
        schema: <object>
} */

in: 'path', 'query' or 'body' // by default is 'query'
description: The parameter description
required: true or false
type: 'string', 'integer', 'object', etc. // by default is 'string'
format: 'int64', etc.
schema: See section Schema and Definitions

For example:

    ...
    app.get('/users/:id', (req, res) => {
        ...
        //  #swagger.parameters['id'] = { description: "User ID" } 
        ...
    })

    app.post('/users', (req, res) => {
        ...
        /*  #swagger.parameters['obj'] = {
                in: 'body',
                type: "object",
                description: "User data"
        } */
        users.addUser(req.body)
        ...
    })

See an example here!

Responses

It is possible to create or complement automatically detected responses. Use the #swagger.reponses[statusCode] tag to create a new answer or to complete an existing answer (automatically detected), for example:

    ...
    app.get('/users/:id', (req, res) => {
        ...
        if(...) {
            /* #swagger.responses[200] = { 
                    description: 'User successfully obtained.',
                    schema: { $ref: "#/definitions/User" } 
            } */
            return res.status(200).send(data)
        }
        ...
        return res.status(404).send(false)
    })

    app.post('/v2/users', (req, res) => {
        ...
        if(...){
            // #swagger.responses[500] = { description: 'Problem with the server.' }
            return res.status(500)
        }     
        ...
    })

NOTE: For more information about schema and definitions, see the section: Schema and Definitions

NOTE: As the 404 status description was not entered, "Not Found" will automatically be added. It is possible to change the language of the automatic response, see the Response Language section.

See an example here!

Schema and Definitions

Unlike how Swagger writes, the answers in this module are added in a simpler way, that is, in the way you want to see the result. These responses can be added to the definitions parameter of the doc object seen in the Usage section, or directly to the response via the schema parameter.

About Examples and Types in schema: The example comes right in front of the parameter declaration, and the type is abstracted according to the typeof of the example. In the code below, the parameter "name" will have as an example "Jhon Doe" and type string, while "age" will have as an example 29 and type number.

NOTE: To configure a parameter as required, just add the symbol $ before the parameter, for example: $name = "Jhon Doe".

const doc = {
    // { ... },
    definitions: {
        Parents: {
            father: "Simon Doe",
            mother: "Marie Doe"
        },
        User: {
            name: "Jhon Doe",
            age: 29,
            parents: {
                $ref: '#/definitions/Parents'
            },
            diplomas: [
                {
                    school: "XYZ University",
                    year: 2020,
                    completed: true,
                    internship: {
                        hours: 290,
                        location: "XYZ Company"
                    }
                }
            ]
        },
        AddUser: {
            $name: "Jhon Doe",
            $age: 29,
            about: ""
        },
       // { ... }
    }
}

Endpoint file:

    app.post('/users', (req, res) => {
        ...
        /*    #swagger.parameters['obj'] = { 
                in: 'body',
                description: "User data.",
                schema: { $ref: "#/definitions/AddUser" }
        } */
        ...
    })

or inserting directly, without using definitions:

    app.post('/users', (req, res) => {
        ...
        /*    #swagger.parameters['obj'] = { 
                in: 'body',
                description: "Adding new user.",
                schema: {
                    $name: "Jhon Doe",
                    $age: 29,
                    about: ""
                }
        } */
        ...
    })

Examples of Definitions

The following are some examples of definitions:

Definitions:

const doc = {
    // { ... },
    definitions: {
        myBoolean: true,
        myNumber: 123,
        myString: "my example",
        myObject: {
            field: "my example"
        },
        myArrayOfBooleans: [ 
            true 
        ],
        myArrayOfNumbers: [ 
            123 
        ],
        myArrayOfStrings: [ 
            "my example" 
        ],
        myArrayOfObjects: [ 
            {
                field: "my example"
            }
        ],
        myReferencedObjectArray: [
            { $ref: "#/definitions/myObject" }
        ]
    }
}

Endpoint:

app.get('/responses', (req, res) => {
  /* #swagger.responses[001] = {
      description: "myBoolean",
      schema: { $ref: "#/definitions/myBoolean" }
  } */

  /* #swagger.responses[002] = {
      description: "myNumber",
      schema: { $ref: "#/definitions/myNumber" }
  } */

  /* #swagger.responses[003] = {
      description: "myString",
      schema: { $ref: "#/definitions/myString" }
  } */

  /* #swagger.responses[004] = {
      description: "myObject",
      schema: { $ref: "#/definitions/myObject" }
  } */

  /* #swagger.responses[005] = {
      description: "myArrayOfBooleans",
      schema: { $ref: "#/definitions/myArrayOfBooleans" }
  } */

  /* #swagger.responses[006] = {
      description: "myArrayOfNumbers",
      schema: { $ref: "#/definitions/myArrayOfNumbers" }
  } */

  /* #swagger.responses[007] = {
      description: "myArrayOfStrings",
      schema: { $ref: "#/definitions/myArrayOfStrings" }
  } */

  /* #swagger.responses[008] = {
      description: "myArrayOfObjects",
      schema: { $ref: "#/definitions/myArrayOfObjects" }
  } */

  /* #swagger.responses[009] = {
      description: "myReferencedObjectArray",
      schema: { $ref: "#/definitions/myReferencedObjectArray" }
  } */
})

The result will be:

See an example here!

Endpoint as deprecated

Use the #swagger.deprecated = true tag to inform that a given endpoint is depreciated, for example:

    ...
    app.get('/users/:id', (req, res) => {
        ...
        // #swagger.deprecated = true
        ...
    })

See an example here!

Ignoring endpoint

Use the #swagger.ignore = true tag to ignore a given endpoint. Thus, it will not appear in the documentation, for example:

    ...
    app.get('/users/:id', (req, res) => {
        ...
        // #swagger.ignore = true
        ...
    })

See an example here!

Manual capture

Use the #swagger.auto = false tag to disable automatic recognition. With that, all parameters of the endpoint must be informed manually, for example:

    ...
    app.put('/users/:id', (req, res) => {
    ...
        /*  #swagger.auto = false

            #swagger.path = '/users/{id}'
            #swagger.method = 'put'
            #swagger.produces = ["application/json"]
            #swagger.consumes = ["application/json"]

            #swagger.parameters['id'] = {
                in: 'path',
                description: 'User ID.',
                required: true,
                type: 'integer'
            }

            #swagger.parameters['obj'] = {
                in: 'body',
                description: 'User data.',
                required: true,
                type: 'string'
            }
        */
        ...
        if(...) {
            // #swagger.responses[201] = { description: "User registered successfully." }
            return res.status(201).send(data)
        }
        ...
        // #swagger.responses[500] = { description: "Server failure."}
        return res.status(500).send(false)
    })

See an example here!

Forced Endpoint Creation

If you want to forcibly create an endpoint, use the #swagger.start and #swagger.end tags, for example:

function myFunction(param) {
    // #swagger.start
    ...
    /*
        #swagger.path = '/forcedEndpoint/{id}'
        #swagger.method = 'put'
        #swagger.description = 'Forced endpoint.'
        #swagger.produces = ["application/json"]
    */
    ...
    /*  #swagger.parameters['id'] = {
            in: 'path',
            type: 'integer',
            description: 'User ID.' } */
    const dataId = users.getUser(req.params.id)
    ...
    /*  #swagger.parameters['obj'] = {
            in: 'query',
            description: 'User data.',
            schema: { $ref: "#/definitions/AddUser" }
    } */
    const dataObj = users.getUser(req.query.obj)
    ...
    if (...)
        return res.status(200).send(true)    // #swagger.responses[200]
    ...
    return res.status(404).send(false)       // #swagger.responses[404]
    ...
    // #swagger.end
}

See an example here!

Security

It is possible to add security to endpoints. The following are some examples, but a complete approach can be seen on the website swagger.io

API Keys (Token) example

The security example below was taken from the original Swagger documentation.

const doc = {
    // { ... },
    securityDefinitions:{
        apiKeyAuth:{
            type: "apiKey",
            in: "header",       // can be "header", "query" or "cookie"
            name: "X-API-KEY",  // name of the header, query parameter or cookie
            description: "any description..."
        }
    }
}

At the endpoint, add the #swagger.security tag, for example:

    ...
    app.get('/users/:id', (req, res) => {
        ...
        /* #swagger.security = [{
               "apiKeyAuth": []
        }] */
        ...
    })

OAuth2 example

The security example below was taken from the original Swagger documentation.

const doc = {
    // { ... },
    securityDefinitions: {
        oAuthSample: {
            type: "oauth2",
            authorizationUrl: "https://petstore.swagger.io/oauth/authorize",
            flow: "implicit",
            scopes: {
                read_pets: "read your pets",
                write_pets: "modify pets in your account"
            }
        }
    }
}

At the endpoint, add the #swagger.security tag, for example:

    ...
    app.get('/users/:id', (req, res) => {
        ...
        /* #swagger.security = [{
            "oAuthSample": [
                "write_pets",
                "read_pets"
            ]
        }] */
        ...
    })

Response Language

It is possible to change the default language (English) of the description in the automatic response, for example: status code 404, the description will be: 'Not Found'. To change, pass an object with the following parameter:

Portuguese (Brazil)

const swaggerAutogen = require('swagger-autogen')({ language: 'pt-BR' }) 
// In this case, for example, the description of status code 404 will be: 
// 'Não Encontrado'

OR

Chinese (Simplified)

const swaggerAutogen = require('swagger-autogen')({ language: 'zh-CN' }) 
// In this case, for example, the description of status code 404 will be: 
// '未找到'

OR

English (by default)

const swaggerAutogen = require('swagger-autogen')()
// In this case, for example, the description of status code 404 will be: 
// 'Not Found'

For now, the module has only the languages: English, Portuguese (Brazil) and Chinese (Simplified).

Examples

Links to projects that covers the simplest use of this module as well as the most complete use. See the links below:

Example using Router

Example without Router

See the result after construction in the image below:

Compatibility

This module is independent of any framework. For the recognition to be automatic, it is necessary that your framework follow the pattern foo.method(path, callback), where foo is the variable belonging to the server or the route, such as: app, server, route, etc. The method are HTTP methods, such as get, post, put, and so on. If the foo.method(path, callback) pattern is not found in the files, it will be necessary to manually enter the beginning and end of the endpoint using the #swagger.start and #swagger.end tags (see the section: Forced Endpoint Creation). If you use the Express.js framework, the status code and produces will be automaticaly obtained according to the status() and setHeader() functions, respectively. If you use a framework that does not contain these functions, you will need to manually add them with the #swagger.response[statusCode] and #swagger.produces tags (see the Responses and Consumes and Produces sections).

Swagger version: 2.0

Tutorials

Some tutorials with examples:

Tutorial in English

Tutorial em Português Brasil

Changelog

  • Version 2.0.x:
    • Recognizes of Routes and referenced functions
    • Endpoint with referenced callback now it's done automatically
    • Multiple patterns now it's done automatically
    • Partial TypeScript recognition
    • Recognizes middleware and middleware array
    • Code refactoring
    • Bug fix
  • Version 2.1.x:
    • Recognizes different file import patterns
    • Recognizes some more features of TypeScript
    • Bug fix
  • Version 2.2.x:
    • Recognizes some more features of TypeScript
    • Performance improvement
    • Recognizes regex in endpoint's path
    • Recognizes middlewares of routes (partially)
    • Options to disable logs
    • Bug fix
  • Version 2.3.x:
    • Recognizes 'require-dir' lib (partially)
    • Recognizes some more features of TypeScript
    • Bug fix
  • Version 2.4.x:
    • Recognizes direct import, such as: router.use(..., require('./routes.js'))
    • Recognizes new Router({ prefix: '...' })
    • Added some default parameters values
    • Code refactoring
    • Bug fix
  • Version 2.5.x:
    • New features: #swagger.summary and #swagger.operationId
    • Bug fix
  • Version 2.6.x:
    • Recognition of more patterns
    • Bug fix

TODO:

  • Recognize middlewares of routes (completely)
  • Recognize regex in middlewares
  • Recognize multiples "express.Router()" in the same file"
  • Recognize referenced path, such as: foo.method(`/${fooPath1}` + '/' + fooPath2, foo...)
  • Recognize 'alias' in the import files
  • Write more test cases
  • Improve performance
  • Refactor code
  • Integrate with other frameworks

Help us!

Help us improve this module. If you have any information that the module does not provide or provides incompletely or incorrectly, please use our Github repository.

pt-Br: Ajude-nos a melhorar este módulo. Se você tiver alguma informação que o módulo não forneça ou forneça de maneira incompleta ou incorreta, use o nosso repositório do Github. Pode enviar em português Brasil também! :)

Repository: https://github.com/davibaltar/swagger-autogen

License

MIT License

swagger-autogen's People

Contributors

davibaltar avatar

Watchers

James Cloos avatar

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.