In order to have someone who will actually use your API, you need to provide some documentation and usages for it. Developer eXperience is also very important when making an API. Making back-end is difficult but having a great front-end too, that is why using your API should be a breeze.
I really like to have my documentation as near as my code, but I want it to be decoupled too. There’s different libraries that offers such a feature, some might use controller annotations like ExDoc, others generate the documentation from the test cases using Bureaucrat, personally I prefer to add my documentation explicitly manually for a matter of separation of concerns using PhoenixSwagger. This way I can eventually not provide external documentation for some endpoints, and have more test cases. It’s a matter of taste.
For the sake of standardisation I prefer to use the OpenAPI Specification (aka Swagger) in order to document my API, it provides some great tooling and is widely supported. That is why I used the phoenix_swagger library and added the documentation in my controllers. You can have a look at the OpenAPI 2.0 Specs to know more about supported formats, parameters, headers and authentication. In Phoenix we have a DSL allowing us to generate the swagger.json file based on our definitions.
To make it work just follow the phoenix_swagger installation guide, you’ll mainly need to provide your router and endpoint module name in the config.ex and a definition of swagger_info/0 in your router
def swagger_info do %{ info: %{ version: "1.0", title: "MY API", consumes: [ "multipart/form-data", "application/json", "application/vnd.api+json" ] }, securityDefinitions: %{ bearerAuth: %{ type: "apiKey", name: "Authorization", in: "header" } }, security: [ %{bearerAuth: []} ] } end
This is mine, you can see that you need to define all of your API supported content-type. I’ve also defined that my API ask for a bearer authorization token.
I’ve also added a route to access the generated API at /api/swagger. It will allow developers to access the last available version of the documentation easily :
scope "/api/swagger" do pipe_through([:api_doc_auth]) forward("/", PhoenixSwagger.Plug.SwaggerUI, otp_app: :MY_API, swagger_file: "swagger.json") end
You can see that I have a pipeline applied to this route, this is because we want to add a basic authentication to the route.
pipeline :api_doc_auth do plug(BasicAuth, use_config: {:gi_api, :api_doc_auth}) end
We used the basic_auth Phoenix plug to make this possible and easy, that’s how you can configure it to use environment variables as login / password for your API documentation route.
config :gi_api, api_doc_auth: [ username: System.get_env("BASIC_AUTH_API_USERNAME"), password: System.get_env("BASIC_AUTH_API_PASSWORD"), realm: "API Doc Area" ]
Now that you’ve configured how you documentation will be available, you just need to declare it :) This is done in your controller, you’ll need to call swagger_path/2 in order to define each endpoint, they can then refer to more complexes schemas that needs to be loaded calling swagger_definitions/0.
swagger_path :index do PhoenixSwagger.Path.get("/api/datas") consumes("application/vnd.api+json") produces("application/vnd.api+json") operation_id("index") tag("Data") paging(size: "page[page_size]", number: "page[page]") description("List data") response(200, "Success", Schema.ref(:Data)) response(401, "Not Authenticated") end
This is my definition for the paginated data schema as a JSON-API resource.
%{ DataResource: JsonApi.resource do description("A data.") relationship(:user) relationship(:media, type: :has_many) attributes do some_attribute(:string, "Data attribute") end end, Data: JsonApi.single(:DataResource), Datas: JsonApi.page(:DataResource) }
On our project we use JsonApi through JaSerializer and pagination thanks to Scrivener. As it is a common stack PhoenixSwagger helpers around JSON-API resources. Very useful when you’re using it as your Data Transfer Protocol. You can define a single resource with JsonApi.single/1 or a paginated resource (based on the paging parameter) with JsonApi.page/1.
Another example with a POST request getting a file from a multipart/form-data form could be.
swagger_path :create do PhoenixSwagger.Path.post("/api/datas/{data_id}/medias") consumes("multipart/form-data") produces("application/json") operation_id("create") tag("Medias") description("Create a media") parameters do data_id(:path, :string, "Data UUID", required: true) kind(:formData, :string, "Should be [image|document]", required: true) file(:formData, :file, "Attached media", required: true) end response(200, "Success", Schema.ref(:Media)) response(404, "Not Found") response(422, "Unprocessable Entity") response(401, "Not Authenticated") end
The :formData specify to Swagger that the API accepts a form formatted field and the type :file means that we will have a binary field and expect a multipart/form-data content type.
With all this you’ll have a smooth API documentation available at /api/swagger, protected by a login / password which supports JWT token authorization header.
You can go further by looking at the PhoenixSwagger documentation. The documentation is not as up to date as I thought but you can access the @doc annotations directly in the source code to have some more examples. I had to find in passed issues too for some of my needs. Besides it also offer easier controller testing through schema validation, but this is another story :)
I hope that this will help you provide a great developer experience to your front end developers and that they will let you waste more time on 9gag now :)
If you like this Elixir / Phoenix blog post serie please share it or drop a comment.
I hope that this will help you provide a great developer experience to your front end developers and that they will let you waste more time on 9gag now :)
If you like this Elixir / Phoenix blog post serie please share it or drop a comment.