Skip to content

Latest commit

 

History

History
447 lines (298 loc) · 18.4 KB

README.md

File metadata and controls

447 lines (298 loc) · 18.4 KB

liblab Llama store

Linting: pylint, Code style: black Test with pytestCheck spec

A llama at the checkout of a store

This is a sample API project to demonstrate how liblab can generate better SDKs for your APIs. This API has a well-formed OpenAPI spec that generates high quality SDKs.

Run the API

Install the dependencies

This API is written in Python using FastAPI. To run it, you'll need Python 3.9 or later. You can install the dependencies using pip, ideally inside a virtual environment:

python -m venv .venv
source .venv/bin/activate
pip install -r requirements.txt

This repo also has a devContainer file, so you can also open it using the dev container in VS Code, GitHub codespaces or other compatible IDE. If you do this, the environment will be fully configured for you.

Prime the database

Before you can run the API you need to configure the SQLite database that is used to store the llamas. This database needs to be primed with some llama data for 6 llamas, as well as creating some pictures. You can create the database using the recreate-database.sh script in the scripts folder:

cd scripts
./recreate-database.sh

This will create a database called sql_app.db in the llama_store/.appdata folder. It will add the following tables to this database:

Table Description
users Contains the users
secrets Contains the secret key used to sign JWT tokens
llamas Contains all the llama
llama_picture_locations Contains the locations on disk of the pictures of the llamas

The script will also create a folder called llama_store/.appdata/llama_store_data/pictures and populate it with pictures of the initial llamas.

If you use the dev container, then this step will be run for your automatically.

Launch the API app

To run the API, navigate to the llama_store folder and run the main.py file:

cd llama_store
python main.py

You can also run this from the command line using the uvicorn command:

uvicorn main:app --reload

Either way will launch the API on localhost on port 8080. You can then navigate to http://localhost:8080/docs to see the Swagger UI for the API. You can change the port number by passing the --port parameter to uvicorn:

uvicorn main:app --reload --port 80

Run the endpoint as readonly

If you want to run this endpoint with the llamas as readonly (for example when hosting publicly and you don't want nefarious members of the public uploading pictures of llamas), you can set the ALLOW_WRITE environment variable to false.

Debug endpoints

This API also can also support GETs on the /user endpoint to list the users when debugging. To turn this on, set the DEBUG environment variable to true. DO NOT do this in production.

Run the API in a Docker container

The API can also be run in a Docker container. To do this, you need to build the container image. On x86/x64 platforms run:

docker buildx build -t llama-store .

On ARM64 (such as macOS on Apple Silicon), run the following:

docker buildx build --platform=linux/arm64 -t llama-store .

You can then run the container. On x86/x64 platforms run:

docker run -p 80:8080 llama-store

On ARM64 (such as macOS on Apple Silicon), run the following:

docker run --platform=linux/arm64 -p 8080:80 llama-store

This will run on port 8080. Change the port number if you want to run it on a different port. The Docker container exposes port 80, but this run command maps it to port 8080 on the host to be consistent with the default uvicorn command.

API end points

This API has the following end points:

Endpoint Description
/openapi.json The OpenAPI spec for the API as a JSON document
/openapi.yaml The OpenAPI spec for the API as a YAML document
/docs The Swagger UI for the API
/redoc The ReDoc UI for the API
/user Register and get a user. You need an access token to get your user.
/token Get a JWT token for a user
/llama Create, read, update, or delete llamas. You need an access token to use this endpoint.
/llama/{llama_id}/pictures Create, read, update, or delete a picture for a llama. You need an access token to use this endpoint.

You can read more about each endpoint in the Swagger UI or ReDoc UI by accessing the /docs or /redoc endpoints from your browser.

Register and create an access tokens

To access the API, you need to have a valid user, and an access token. This API uses JWT tokens, signed with a secret key. The secret key is stored in the database, and is created when the database is created. The secret key is not exposed by the API. These tokens are valid for 30 minutes, and are bearer tokens passed in the Authorization header, with the value Bearer <access_token>.

To create a user, send a POST request to the /user endpoint, passing in a JSON body with the following format:

{
  "email": "string",
  "password": "string"
}

The email is validated to ensure it is a valid email address, and the password must be at least 8 characters long, with one upper case letter, one lower case letter, one number, and one special character.

This endpoint does not require an access token.

Once your user has been created, you can request an access token by sending a POST request to the /token endpoint, passing in a JSON body with the following format:

{
  "email": "string",
  "password": "string"
}

This needs to match the user you created. If the user is valid, you will get a response with a JSON body with the following format:

{
  "access_token": "string",
  "token_type": "string"
}

Once you have the access token, you can use it to access the other endpoints. To do this, you need to pass the access token in the Authorization header, with the value Bearer <access_token>.

Generate an SDK

Once your API is running, you can generate an SDK. Start by installing liblab:

npm install -g liblab

You will need to register or login using:

liblab login

If you don't have an account - join our beta.

You can learn more about how to use liblab from our developer docs.

The liblab CLI uses a config file called liblab.config.json to configure the SDK. This repo has a config file called liblab.config.json that you can use to generate the SDK.

This config file reads the local spec.json file. If you want to generate an SDK from a running API, you can change this to the URL of that API. SDKs will be generated for Java, Python and TypeScript with a name of llama-store (adjust to be language specific, so llamaStore in Java and TypeScript). The SDKs will be configured to use bearer tokens for authentication, and will include documentation. The generated SDKs will also be set up with dev containers for VS Code, so you can open the created SDK folder and get going straight away.

To generate the SDKs, run the following command:

liblab build

The SDKs will be created and downloaded to the output folder. You can then use these SDKs in your applications.

SDK docs will also be created, and you will be able to access them online, or download them.

Pre-built SDKs

You can find pre-built SDKs in the following GitHub repos:

Language Repo
Python llama-store-sdk-python
Java llama-store-sdk-java
TypeScript llama-store-sdk-typescript
C# llama-store-sdk-csharp
Go llama-store-sdk-go

These are generated by a GitHub action, and are updated whenever the spec changes. You can find the publish_sdk.yaml action in the .github/workflows folder.

This is the action:

name: Publish SDK

on:
  push:
    paths:
      - 'spec.json'
  workflow_dispatch:

jobs:
  build-and-pr:
    name: Generate SDKs and create PRs
    runs-on: ubuntu-latest
    env:
      LIBLAB_TOKEN: ${{ secrets.LIBLAB_TOKEN }}
      GITHUB_TOKEN: ${{ secrets.LIBLAB_GITHUB_TOKEN }}
    steps:
      - name: Checkout
        uses: actions/checkout@v3
      - name: Setup Node.js environment
        uses: actions/setup-node@v3
        with:
          node-version: "18" # Specify the node version you want to use
      - name: Install liblab
        run: npm install -g liblab
      - name: Start Build
        run: liblab build --skip-validation --approve-docs
      - name: Create PRs to GitHub repos
        run: liblab pr

You can read more about the process to publish an SDK in an action in our developer docs.

Run the SDK examples

There are SDK examples for TypeScript and Python in the sdk-examples folder.

Before running these examples, you will need to have the llama store running with a clean database. For best results, use the dev container in this repo:

  1. Ensure you have Docker running
  2. Ensure you have the VS Code remote development extension pack installed
  3. Open this repo in VS Code, and when prompted re-open in the container

This will install everything you need.

Next you will need to build the SDKs:

  1. Run liblab login to login (the liblab CLI is installed as part of this dev container).
  2. Run liblab build from the root of this repo. This will build against a saved version of the spec.

Next, you need to launch the Llama store:

  1. From a terminal, run:

    ./scripts/start-llama-store.sh

    This will reset the llama store database, then launch the API on port 8080.

Once you have done this, you can run the examples. You will need to create a new terminal to do this.

TypeScript

To run the TypeScript examples, navigate to the sdk-examples/typescript folder.

  1. Build the SDK with the following command from that folder:

    npm run setup
  2. Run the get llamas demo with the following command:

    npm run get-llamas

    This will create a user, generate an API token, and print out a list of llamas. This demo shows the ability to call services on the SDK, set an API token once, and use that for all subsequent calls.

  3. Run the create llamas demo with the following command:

    npm run create-llamas

    This will create a user, generate an API token, and create a llama.

  4. Run the get llamas demo again with the following command:

    npm run get-llamas

    You will see the llama you created in the previous step in the list of llamas.

Python

To run the Python examples, navigate to the sdk-examples/python folder.

  1. Build and install the Python SDK with the following command from that folder:

    ./setup-python.sh

    This assumes you are running in a dev container as it doesn't create a virtual environment. If you plan to run this locally, you should create and activate a virtual environment first.

  2. Run the get llamas demo with the following command:

    python get_llamas.py

    This will create a user, generate an API token, and print out a list of llamas. It will also download pictures for all the llamas into the pics folder. This demo shows the ability to call services on the SDK, set an API token once, and use that for all subsequent calls.

  3. Run the create llamas demo with the following command:

    python create_llamas.py

    This will create a user, generate an API token, and create a llama, uploading a picture.

  4. Run the get llamas demo again with the following command:

    python get_llamas.py

    You will see the llama you created in the previous step in the list of llamas.

Go

To run the Go examples, you will need to copy the contents of the sdk-examples/go folder into the output/go/cmd/examples folder.

  1. Run the get llamas demo with the following command:

    go run get-llamas.go

    This will create a user, generate an API token, and print out a list of llamas. This demo shows the ability to call services on the SDK, set an API token once, and use that for all subsequent calls.

  2. Run the create llamas demo with the following command:

    go run create-llama.go

    This will create a user, generate an API token, and create a llama, uploading a picture.

  3. Run the get llamas demo again with the following command:

    go run get-llamas.go

    You will see the llama you created in the previous step in the list of llamas.

OpenAPI spec

The OpenAPI spec for this API is in the spec.json and spec.yaml files. These need to be generated whenever the spec changes. To do this, run the following command:

cd scripts
./create-specs.sh

Generate API documentation

As a part of the SDK generation using the provided liblab config file, the SDKs for some languages will include generated code snippets.

These snippets are designed to be included in your own developer documentation. You can see an example of this in the docs folder. This is a Docusaurus site that uses the Docusaurus OpenAPI plugin to generate API docs, with an updated theme component showing how to include SDK snippets generated by liblab.

Generate the documentation

To generate the documentation, you need to do the following:

  1. Generate the SDKs using liblab

  2. Copy the generated snippets from the SDK folders, to the docs/snippets folder, renaming each one to match the language it is for. For example, copy output/csharp/Documentation/snippets.json to docs/snippets/csharp.json.

  3. Head to the docs folder, and generate the API docs. This uses the Docusaurus OpenAPI plugin:

    cd docs
    npm install
    npm run docusaurus clean-api-docs all
    npm run docusaurus gen-api-docs all

There is a helper script that does all of this for you. Run the following command from the root of this repo:

./scripts/create-sdk-docs.sh

Launch the documentation

To launch the documentation, you need to do the following:

  1. Navigate to the docs folder:

    cd docs
  2. Start the Docusaurus site:

    npm run start

You can then open the documentation in your browser at http://localhost:3000. You will be greated by this readme rendered as docs, with the API reference on the side. Expand the API reference to see the full API documentation for each endpoint, with SDK snippets.

A screenshot of the llama store docs showing the Create API token endpoint and example C# code

How SDK snippets work

To show the SDK snippets, the Docusaurus OpenAPI theme was swizzled to wrap the APIExplorer/CodeSnippets component. In the default theme, this component shows examples of calling the API directly in different programming languages.

The swizzled version loads the SDK snippets created with your SDK, and shows those instead. You can see this component at docs/src/theme/APIExplorer/CodeSnippets/index.tsx.

The SDK snippets are in a JSON file that lists the snippets by endpoint and method:

{
  "endpoints": {
    "/llama": {
      "get": "from llama_store import LlamaStore, Environment\n\nsdk = LlamaStore(\n    access_token=\"YOUR_ACCESS_TOKEN\",\n    base_url=Environment.DEFAULT.value\n)\n\nresult = sdk.llama.get_llamas()\n\nprint(result)\n",
      "post": "from llama_store import LlamaStore, Environment\nfrom llama_store.models import LlamaCreate\n\nsdk = LlamaStore(\n    access_token=\"YOUR_ACCESS_TOKEN\",\n    base_url=Environment.DEFAULT.value\n)\n\nrequest_body = LlamaCreate(\n    name=\"libby the llama\",\n    age=5,\n    color=\"brown\",\n    rating=1\n)\n\nresult = sdk.llama.create_llama(request_body=request_body)\n\nprint(result)\n"
    },
    "/llama/{llama_id}": {
      "get": "from llama_store import LlamaStore, Environment\n\nsdk = LlamaStore(\n    access_token=\"YOUR_ACCESS_TOKEN\",\n    base_url=Environment.DEFAULT.value\n)\n\nresult = sdk.llama.get_llama_by_id(llama_id=\"1\")\n\nprint(result)\n",
      "put": "from llama_store import LlamaStore, Environment\nfrom llama_store.models import LlamaCreate\n\nsdk = LlamaStore(\n    access_token=\"YOUR_ACCESS_TOKEN\",\n    base_url=Environment.DEFAULT.value\n)\n\nrequest_body = LlamaCreate(\n    name=\"libby the llama\",\n    age=5,\n    color=\"brown\",\n    rating=1\n)\n\nresult = sdk.llama.update_llama(\n    request_body=request_body,\n    llama_id=\"1\"\n)\n\nprint(result)\n",
      "delete": "from llama_store import LlamaStore, Environment\n\nsdk = LlamaStore(\n    access_token=\"YOUR_ACCESS_TOKEN\",\n    base_url=Environment.DEFAULT.value\n)\n\nresult = sdk.llama.delete_llama(llama_id=\"1\")\n\nprint(result)\n"
    },
    ...
  }
  ...
}

Inside the component, the JSON file is loaded, and using the current API endpoint and method, the relevant snippet is extracted from the JSON and shown in an ApiCodeBlock - a component from the theme that shows formatted code.

Check out the component source code for more details on how this works.