# Community and Support Get help with Hypermode and connect with other developers Whether you're just getting started or you're a seasoned developer, we're here to help you every step of the way. We’re excited to have you as part of the Hypermode community and look forward to seeing what you build! ## Community [Discord](https://discord.hypermode.com) is our main forum where you can ask questions, share your knowledge, and connect with other developers. ## Getting help If you encounter a bug or have a feature request, you can open an issue on the relevant GitHub repository: * [Modus](https://github.com/hypermodeinc/modus/issues) * [Hyp CLI](https://github.com/hypermodeinc/hyp-cli/issues) All paid Hypermode packages include commercial support. Customers can reach out via the Hypermode Console or through email at [help@hypermode.com](mailto:help@hypermode.com). ## Stay connected Stay up-to-date with the latest news, updates, and announcements from Hypermode: * **[X](https://x.com/hypermodeinc)**: follow us on X for the latest news and updates * **[Blog](https://hypermode.com/blog)**: explore our blog for in-depth articles, tutorials, and case studies # Configure Environment Define environment parameters for your app On your first deployment, and when you add new connections thereafter, you may need to configure your environment for your app to be ready for traffic. ## Connection secrets If you included parameters for Connection Secrets in your [app manifest](/modus/app-manifest), you'll need to add the parameter values in the Hypermode Console. From your project home, navigate to **Settings** → **Connections**. Add the values for your defined connection authentication parameters. ## Model hosting Where available, Hypermode defaults to shared model instances. Refer to the [Hosted Models](/hosted-models) documentation for instructions on setting up hosted models. ## Scaling your runtime resources We're working to make runtime scaling self-service. In the meantime, reach out at [help@hypermode.com](mailto:help@hypermode.com) for assistance with this request. # Create Project Initialize your Modus app with Hypermode A Hypermode project represents your Modus app and associated models. You can create a project through the Hypermode Console or Hyp CLI. ## Create with Hypermode Console Hypermode relies on Git as a source for project deployment. Through the project creation flow, you'll connect Hypermode to your GitHub account. From your organization home, click **New Project**. Set a name for the project and click **Create**. Once you have created a project, Hypermode prompts you to finish setting up your project using the CLI. First, install the Modus CLI and initialize your app. For more information on creating your first Modus app, visit the [Modus quickstart](modus/quickstart). Next, initialize the app with Hypermode through the [Hyp CLI](/hyp-cli) and link your GitHub repo with your Modus app to Hypermode using: ```bash hyp link ``` This command adds a default GitHub Actions workflow to build your Modus app and a Hypermode GitHub app for auto-deployment. Once initialized, each commit to the target branch triggers a deployment. You can also connect to an existing GitHub repository. The last step is triggering your first deployment. If you linked your project through the Hyp CLI, make sure to push your changes first to trigger a deployment. # Deploy Project A git-based flow for simple deployment Hypermode features a native GitHub integration for the deployment of Hypermode projects. The deployment includes your Modus app as well as any models defined in the [app manifest](/modus/app-manifest). Preview environments for live validation of pull requests are in development. ## Link your project to GitHub After you push your Modus app to GitHub, you can link your Hypermode project to the repo through the Hyp CLI. ```bash hyp link ``` ## Build When you link your project with Hypermode, the Hyp CLI adds a GitHub Actions workflow to your repo that builds your Modus app to Hypermode on commit. ## Deploy On successful build of your project, Hypermode automatically deploys your project changes. For [hosted models](/hosted-models), Hypermode creates a connection for your app to a shared or dedicated model instance. You can view the deployment status from the Deployments tab within the Hypermode Console. # Hosted Models Iterate quickly with seamless access to the most popular models Hypermode includes a set of shared models available for integration into your app on a pay-per-token basis. Need a bespoke model? You can include a model from [Hugging Face](https://huggingface.co/) in your [app manifest](/modus/app-manifest) and Hypermode runs and manages it for you. ## Setup To use a Hypermode-hosted model, set `connection: "hypermode"`, `provider: "hugging-face"`, and set `sourceModel` to be the model name as specified on Hugging Face. ```json modus.json {3-7} { ... "models": { "text-generator": { "sourceModel": "meta-llama/Meta-Llama-3.1-8B-Instruct", "provider": "hugging-face", "connection": "hypermode" } } ... } ``` ## Deployment mode We run our most popular models as multi-tenant, shared instances across projects and customers. By default, if the model you use is available as a shared model, your app uses these shared models at runtime. If the model you use isn't available as a shared model, Hypermode automatically spins up a dedicated instance of the model for your project. ## Shared models These are the models available currently with shared instances: * [`meta-llama/Meta-Llama-3.1-8B-Instruct`](https://huggingface.co/meta-llama/Meta-Llama-3.1-8B-Instruct) * [`meta-llama/Llama-3.2-3B-Instruct`](https://huggingface.co/meta-llama/Llama-3.2-3B-Instruct) * [`deepseek-ai/DeepSeek-R1-Distill-Llama-8B`](https://huggingface.co/deepseek-ai/DeepSeek-R1-Distill-Llama-8B) * [`sentence-transformers/all-MiniLM-L6-v2`](https://huggingface.co/sentence-transformers/all-MiniLM-L6-v2) * [`AntoineMC/distilbart-mnli-github-issues`](https://huggingface.co/AntoineMC/distilbart-mnli-github-issues) * [`distilbert/distilbert-base-uncased-finetuned-sst-2-english`](https://huggingface.co/distilbert/distilbert-base-uncased-finetuned-sst-2-english) We're constantly evaluating model usage in determining new models to add to our shared catalog. Interested in consuming an open source model not listed here by the token? Let us know at [help@hypermode.com](mailto:help@hypermode.com). # Hyp CLI Comprehensive reference for the Hyp CLI commands and usage Hyp CLI is a command-line tool for managing your Hypermode account and projects. When using Hyp CLI alongside the Modus CLI, users get access to [Hypermode-hosted models](/hosted-models). ## Install Install Hyp CLI via npm. ```bash npm install -g @hypermode/hyp-cli ``` ## Commands ### `login` Log in to your Hypermode account. When executed, this command redirects to the Hypermode website to authenticate. It stores an authentication token locally and prompts you to select your organization context. ### `logout` Log out of your Hypermode account. This command clears your local authentication token. ### `org switch` Switch to a different org context within the CLI session. # Integrate API Easily add intelligent features to your app Hypermode makes it easy to incrementally add intelligence your app. ## API endpoint You can find your project's API endpoint in the Hypermode Console, in your project's Home tab, in the format `https://.hypermode.app/`. ## API token Hypermode protects your project's endpoint with an API key. In your project dashboard, navigate to **Settings** → **API Keys** to find and manage your API tokens. From your app, you can call the API by passing the API token in the `Authorization` header. Here's an example using the `fetch` API in JavaScript: ```javascript const endpoint = "" // your API endpoint const key = "" // your API key // your graphql query const query = `query { ... }` const response = await fetch(endpoint, { method: "POST", headers: { "Content-Type": "application/json", Authorization: key, }, body: JSON.stringify({ query: query, }), }) const data = await response.json() ``` Additional authorization methods are under development. If your app requires a different protocol, reach out at [help@hypermode.com](mailto:help@hypermode.com). # Introduction Build Intelligent APIs. {/* vale Google.Contractions = NO */} ## What is Hypermode? {/* vale Google.Contractions = YES */} Hypermode is a managed service that provides the workbench and infrastructure to create **powerful, engaging, and secure AI assistants, APIs, and backend services**. With every push to Git, Hypermode automatically builds and deploys your functions. It creates a live, scalable API for you to preview, test, and promote to production. ## What's included with Hypermode? Hypermode provides a suite of tools and services to help you build and manage your intelligent APIs. Use one component or all of them together as the complete backend for your app. Modus is an open source, serverless framework for building functions and APIs, powered by WebAssembly Experiment rapidly with access to the most popular and newest models. Dgraph is a distributed, transactional graph database optimized for knowledge graph creation and management End-to-end observability and control for your functions, models, and data. Hypermode's console showing project, model, and API management. # Modify Organization Update organization attributes or delete your organization Organizations contain a set of related projects and members. This forms the billing envelope for Hypermode. You can modify your organization's name and slug through the settings. ## Update organization name Your organization name is a display name only and changing it doesn't impact your running apps. To update your organization display name, click your organization's name in the top navigation. Then, click **Settings** → **General**. Enter a new name and click **Save**. ## Update organization slug From your organization home view, click **Settings** → **General**. Enter a new slug and click **Save**. When you update the organization slug, Hypermode reflects the new slug in your app endpoints across all projects. You must update any existing integrations that use the existing app endpoints. Make sure to communicate these changes with your team or stakeholders. ## Delete organization If you no longer need an organization, you can permanently delete it. To delete an organization, you must [delete all projects](/modify-project#delete-project) first. To delete an organization, click your organization's name in the top navigation. Then, click **Settings** → **Delete**. Click **Delete** and enter the organization's name as confirmation. This action is irreversible and permanently deletes your organization. # Modify Project Update project attributes or delete your project After you create a project, you can modify its name and slug through the project settings. ## Update project name Your project name is display-only and changing it doesn't impact your running app. To update your project display name, click **Settings** → **General**. Enter a new name and click **Save**. ## Update project slug From your project view, click **Settings** → **General**. Enter a new slug and click **Save**. If you update the project slug, you must update any existing integrations that use the existing app endpoints. Make sure to communicate these changes with your team or stakeholders. ## Delete project If you no longer need a project, you can permanently delete it. Deleting a project removes all associated endpoints, configurations, and data. To delete a project, from the project in the console, click **Settings** → **Delete**. Click **Delete** and enter the project’s name as confirmation. This action is irreversible and permanently deletes all project data and configuration. # API Generation Create the signature for your API Modus automatically creates an external API based on the endpoints defined in your [app manifest](/modus/app-manifest#endpoints). Modus generates the API signature based on the functions you export from your app. ## Exporting functions Modus uses the default conventions for each language. Functions written in Go use starting capital letters to expose functions as public. Modus creates an external API for public functions from any file that belongs to the `main` package. The functions below generate an API endpoint with the signature ```graphql type Query { classifyText(text: String!, threshold: Float!): String! } ``` Since the `classify` function isn't capitalized, Modus doesn't include it in the generated GraphQL API. ```go package main import ( "errors" "fmt" "github.com/hypermodeAI/functions-go/pkg/models" "github.com/hypermodeAI/functions-go/pkg/models/experimental" ) const modelName = "my-classifier" // this function takes input text and a probability threshold, and returns the // classification label determined by the model, if the confidence is above the // threshold; otherwise, it returns an empty string func ClassifyText(text string, threshold float32) (string, error) { predictions, err:= classify(text) if err != nil { return "", err } prediction := predictions[0] if prediction.Confidence < threshold { return "", nil } return prediction.Label, nil } func classify(texts ...string) ([]experimental.ClassifierResult, error) { model, err := models.GetModel[experimental.ClassificationModel](modelName) if err != nil { return nil, err } input, err := model.CreateInput(texts...) if err != nil { return nil, err } output, err := model.Invoke(input) if err != nil { return nil, err } if len(output.Predictions) != len(texts) { word := "prediction" if len(texts) > 1 { word += "s" } return nil, fmt.Errorf("expected %d %s, got %d", len(texts), word, len(output.Predictions)) } return output.Predictions, nil } ``` Functions written in AssemblyScript use ES module-style `import` and `export` statements. With the default package configuration, Modus creates an external API for functions exported form the `index.ts` file located in the `functions/assembly` folder of your project. The functions below generate an API endpoint with the signature ```graphql type Query { classifyText(text: String!, threshold: Float!): String! } ``` Since the `classify` function isn't exported from the module, Modus doesn't include it in the generated GraphQL API. ```ts import { models } from "@hypermode/modus-sdk-as" import { ClassificationModel, ClassifierResult, } from "@hypermode/modus-sdk-as/models/experimental/classification" const modelName: string = "my-classifier" // this function takes input text and a probability threshold, and returns the // classification label determined by the model, if the confidence is above the // threshold; otherwise, it returns an empty string export function classifyText(text: string, threshold: f32): string { const predictions = classify(text, threshold) const prediction = predictions[0] if (prediction.confidence < threshold) { return "" } return prediction.label } function classify(text: string, threshold: f32): ClassifierResult[] { const model = models.getModel(modelName) const input = model.createInput([text]) const output = model.invoke(input) return output.predictions } ``` # App Manifest Define the resources for your app The manifest for your Modus app allows you to configure the exposure and resources for your functions at runtime. You define the manifest in the `modus.json` file within the root of directory of your app. ## Structure Expose your functions for integration into your frontend or federated API Establish connectivity for external endpoints and model hosts Define inference services for use in your functions Define sets of text data to enable natural language search ### Base manifest A simple manifest, which exposes a single GraphQL endpoint with a bearer token for authentication, looks like this: ```json modus.json { "$schema": "https://schema.hypermode.com/modus.json", "endpoints": { "default": { "type": "graphql", "path": "/graphql", "auth": "bearer-token" } } } ``` ## Endpoints Endpoints make your functions available outside of your Modus app. The `endpoints` object in the app manifest allows you to define these endpoints for integration into your frontend or federated API. Each endpoint requires a unique name, specified as a key containing only alphanumeric characters and hyphens. Only a GraphQL endpoint is available currently, but the modular design of Modus allows for the introduction of additional endpoint types in the future. ### GraphQL endpoint This endpoint type supports the GraphQL protocol to communicate with external clients. You can use a GraphQL client, such as [urql](https://github.com/urql-graphql/urql) or [Apollo Client](https://github.com/apollographql/apollo-client), to interact with the endpoint. **Example:** ```json modus.json { "endpoints": { "default": { "type": "graphql", "path": "/graphql", "auth": "bearer-token" } } } ``` Always set to `"graphql"` for this endpoint type. The path for the endpoint. Must start with a forward slash `/`. The authentication method for the endpoint. Options are `"bearer-token"` or `"none"`. See [Authentication](/modus/authentication) for additional details. ## Connections Connections establish connectivity and access to external services. They're used for HTTP and GraphQL APIs, database connections, and externally hosted AI models. The `connections` object in the app manifest allows you to define these hosts, for secure access from within a function. Each connection requires a unique name, specified as a key containing only alphanumeric characters and hyphens. Each connection has a `type` property, which controls how it's used and which additional properties are available. The following table lists the available connection types: | Type | Purpose | Function Classes | | :----------- | :------------------------------- | :-------------------------- | | `http` | Connect to an HTTP(S) web server | `http`, `graphql`, `models` | | `dgraph` | Connect to a Dgraph database | `dgraph` | | `mysql` | Connect to a MySQL database | `mysql` | | `neo4j` | Connect to a Neo4j database | `neo4j` | | `postgresql` | Connect to a PostgreSQL database | `postgresql` | **Don't include secrets directly in the manifest!** If your connection requires authentication, you can include *placeholders* in connection properties which resolve to their respective secrets at runtime. When developing locally, [set secrets using environment variables](#working-locally-with-secrets). When deployed on Hypermode, set the actual secrets via the Hypermode Console, where they're securely stored until needed. ### HTTP connection This connection type supports the HTTP and HTTPS protocols to communicate with external hosts. You can use the [HTTP APIs](/modus/sdk/assemblyscript/http) in the Modus SDK to interact with the host. This connection type is also used for [GraphQL APIs](/modus/sdk/assemblyscript/graphql) and to invoke externally hosted AI [models](/modus/sdk/assemblyscript/models). **Example:** ```json modus.json { "connections": { "openai": { "type": "http", "baseUrl": "https://api.openai.com/", "headers": { "Authorization": "Bearer {{API_KEY}}" } } } } ``` Always set to `"http"` for this connection type. Base URL for connections to the host. Must end with a trailing slash and may contain path segments if necessary. Example: `"https://api.example.com/v1/"` Full URL endpoint for connections to the host. Example: `"https://models.example.com/v1/classifier"` You must include either a `baseUrl` or an `endpoint`, but not both. * Use `baseUrl` for connections to a host with a common base URL. * Use `endpoint` for connections to a specific URL. Typically, you'll use the `baseUrl` field. However, some APIs, such as `graphql.execute`, require the full URL in the `endpoint` field. If provided, requests on the connection include these headers. Each key-value pair is a header name and value. Values may include variables using the `{{VARIABLE}}` template syntax, which resolve at runtime to secrets provided for each connection, via the Hypermode Console. This example specifies a header named `Authorization` that uses the `Bearer` scheme. A secret named `AUTH_TOKEN` provides the token: ```json "headers": { "Authorization": "Bearer {{AUTH_TOKEN}}" } ``` This example specifies a header named `X-API-Key` provided by a secret named `API_KEY`: ```json "headers": { "X-API-Key": "{{API_KEY}}" } ``` You can use a special syntax for connections that require [HTTP basic authentication](https://en.wikipedia.org/wiki/Basic_access_authentication). In this example, secrets named `USERNAME` and `PASSWORD` combined and then are base64-encoded to form a compliant `Authorization` header value: ```json "headers": { "Authorization": "Basic {{base64(USERNAME:PASSWORD)}}" } ``` If provided, requests on the connection include these query parameters, appended to the URL. Each key-value pair is a parameter name and value. Values may include variables using the `{{VARIABLE}}` template syntax, which resolve at runtime to secrets provided for each connection, via the Hypermode Console. This example specifies a query parameter named `key` provided by a secret named `API_KEY`: ```json "queryParameters": { "key": "{{API_KEY}}" } ``` ### Dgraph connection This connection type supports connecting to Dgraph databases. You can use the [Dgraph APIs](/modus/sdk/assemblyscript/dgraph) in the Modus SDK to interact with the database. **Example:** ```json modus.json { "connections": { "my-dgraph": { "type": "dgraph", "grpcTarget": "frozen-mango.grpc.eu-central-1.aws.cloud.dgraph.io:443", "key": "{{DGRAPH_API_KEY}}" } } } ``` Always set to `"dgraph"` for this connection type. The gRPC target for the Dgraph database. The API key for the Dgraph database. ### MySQL connection This connection type supports connecting to MySQL databases. You can use the [MySQL APIs](/modus/sdk/assemblyscript/mysql) in the Modus SDK to interact with the database. **Example:** ```json modus.json { "connections": { "my-database": { "type": "mysql", "connString": "mysql://{{USERNAME}}:{{PASSWORD}}@db.example.com:3306/dbname?tls=true" } } } ``` Always set to `"mysql"` for this connection type. The connection string for the MySQL database. Values may include variables using the `{{VARIABLE}}` template syntax, which resolve at runtime to secrets provided for each connection, via the Hypermode Console. The connection string in the preceding example includes: * A username and password provided by secrets named `USERNAME` & `PASSWORD` * A host named `db.example.com` on port `3306` * A database named `dbname` * Encryption enabled via `tls=true` - which is highly recommended for secure connections Set the connection string using a URI format [as described in the MySQL documentation](https://dev.mysql.com/doc/refman/8.4/en/connecting-using-uri-or-key-value-pairs.html#connecting-using-uri). However, any optional parameters provided should be in the form specified by the Go MySQL driver used by the Modus Runtime, [as described here](https://github.com/go-sql-driver/mysql/blob/master/README.md#parameters) For example, use `tls=true` to enable encryption (not `sslmode=require`). ### Neo4j connection This connection type supports connecting to Neo4j databases. You can use the [Neo4j APIs](/modus/sdk/assemblyscript/neo4j) in the Modus SDK to interact with the database. **Example:** ```json modus.json { "connections": { "my-neo4j": { "type": "neo4j", "dbUri": "bolt://localhost:7687", "username": "neo4j", "password": "{{NEO4J_PASSWORD}}" } } } ``` Always set to `"neo4j"` for this connection type. The URI for the Neo4j database. The username for the Neo4j database. The password for the Neo4j database. ### PostgreSQL connection This connection type supports connecting to PostgreSQL databases. You can use the [PostgreSQL APIs](/modus/sdk/assemblyscript/postgresql) in the Modus SDK to interact with the database. **Example:** ```json modus.json { "connections": { "my-database": { "type": "postgresql", "connString": "postgresql://{{PG_USER}}:{{PG_PASSWORD}}@db.example.com:5432/data?sslmode=require" } } } ``` Always set to `"postgresql"` for this connection type. The connection string for the PostgreSQL database. Values may include variables using the `{{VARIABLE}}` template syntax, which resolve at runtime to secrets provided for each connection, via the Hypermode Console. The connection string in the preceding example includes: * A username and password provided by secrets named `PG_USER` & `PG_PASSWORD` * A host named `db.example.com` on port `5432` * A database named `data` * SSL mode set to `require` - which is highly recommended for secure connections Refer to [the PostgreSQL documentation](https://www.postgresql.org/docs/current/libpq-connect.html#LIBPQ-CONNSTRING) for more details on connection strings. Managed PostgreSQL providers often provide a pre-made connection string for you to copy. Check your provider's documentation for details. For example, if using Neon, refer to the [Neon documentation](https://neon.tech/docs/connect/connect-from-any-app). ### Working locally with secrets When you run your app locally using `modus dev`, the runtime replaces the placeholders of the manifest with values from environment variables defined in your operating system or in `.env` files. The environment variables keys must be upper case and follow the naming convention: `MODUS__` For example, with the following manifest: ```json modus.json { "connections": { "openai": { "type": "http", "baseUrl": "https://api.openai.com/", "headers": { "Authorization": "Bearer {{API_KEY}}" } } } } ``` The Modus runtime substitutes `{{API_KEY}}` with the value of the environment variable `MODUS_OPENAI_API_KEY` An easy way to define the environment variables when working locally is to use the file `.env.dev.local` located in your app folder. For the previous manifest, we can set the key in the .env.dev.local file as follow: ```text .env.dev.local MODUS_OPENAI_API_KEY="your openai key" ``` You should exclude `.env` files from source control. Projects created with `modus new` exclude these files automatically when creating your project. ## Models AI models are a core resource for inferencing. The `models` object in the app manifest allows you to easily define models, whether hosted by Hypermode or another host. Each model requires a unique name, specified as a key, containing only alphanumeric characters and hyphens. ```json modus.json { "models": { "text-generator": { "sourceModel": "meta-llama/Meta-Llama-3.1-8B-Instruct", "provider": "hugging-face", "connection": "hypermode" } } } ``` Original relative path of the model within the provider's repository. Source provider of the model. If the `connection` value is `hypermode`, this field is mandatory. `hugging-face` is currently the only supported option. Connection for the model instance. * Specify `"hypermode"` for models that [Hypermode hosts](/hosted-models). * Otherwise, specify a name that matches a connection defined in the [`connections`](#connections) section of the manifest. When using `hugging-face` as the `provider` and `hypermode` as the `connection`, Hypermode automatically facilitates the connection to an instance of a shared or dedicated instance of the model. Your project's functions securely access the hosted model, with no further configuration required. For more details, see [hosted models](/hosted-models). ## Collections Collections simplify the usage of vector embeddings to build natural language search features. The `collections` object allows you to define indexed data types that are automatically embedded and searchable based on the search method you define. Each collection requires a unique name, specified as a key, containing only alphanumeric characters and hyphens. For more detail on implementing Collections, see [Search](/modus/search). ```json modus.json { "collections": { "myProducts": { "searchMethods": { "searchMethod": { "embedder": "myEmbedder", "index": { "type": "sequential" } } } } } } ``` Search methods define a pair of an embedder and index to make available for searching the data in your collection. The function name to embed text added to the collection. If provided, describes the index mechanism used by the search method. `type`: specifies the type of the index. For example, `sequential` (default). # Authentication Protect your API It's easy to secure your Modus app with authentication. Modus currently supports bearer token authentication, with additional authentication methods coming soon. ## Bearer tokens Modus supports authentication via the `Authorization` header in HTTP requests. You can use the `Authorization` header to pass a bearer JSON Web Token (JWT) to your Modus app. The token authenticates the user and authorize access to resources. To use bearer token authentication for your Modus app, be sure to set the `auth` property on your endpoint to `"bearer-token"` in your [app manifest](/modus/app-manifest#endpoints). ### Setting verification keys Once set, Modus verifies tokens passed in the `Authorization` header of incoming requests against the public keys you provide. To enable this verification, you must pass the public keys using the `MODUS_PEMS` or `MODUS_JWKS_ENDPOINTS` environment variable. The value of the `MODUS_PEMS` or `MODUS_JWKS_ENDPOINTS` environment variable should be a JSON object with the public keys as key-value pairs. This is an example of how to set the `MODUS_PEMS` and `MODUS_JWKS_ENDPOINTS` environment variable: ```bash MODUS_PEMS export MODUS_PEMS='{\"key1\":\"-----BEGIN PUBLIC KEY-----\\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAwJ9z1z1z1z1z1z\\n-----END PUBLIC KEY-----\"}' ``` ```bash MODUS_JWKS_ENDPOINTS export MODUS_JWKS_ENDPOINTS='{"my-auth-provider":"https://myauthprovider.com/application/o/myappname/.wellknown/jwks.json"}' ``` When deploying your Modus app on Hypermode, the bearer token authentication is automatically set up. ### Verifying tokens To verify the token, Modus uses the public keys passed via the `MODUS_PEMS` environment variable. If the token is verifiable with any of the verification keys provided, Modus decodes the JWT token and passes the decoded claims as an environment variable. ### Accessing claims The decoded claims are available through the `auth` API in the Modus SDK. To access the decoded claims, use the `getJWTClaims()` function. The function allows the user to pass in a class to deserialize the claims into, and returns an instance of the class with the claims. This allows users to access the claims in the token and use them to authenticate and authorize users in their Modus app. ```go Go import github.com/hypermodeinc/modus/sdk/go/pkg/auth type ExampleClaims struct { Sub string `json:"sub"` Exp int64 `json:"exp"` Iat int64 `json:"iat"` } func GetClaims() (*ExampleClaims, error) { return auth.GetJWTClaims[*ExampleClaims]() } ``` ```ts AssemblyScript import { auth } from "@hypermode/modus-sdk-as" @json export class ExampleClaims { public sub!: string public exp!: i64 public iat!: i64 } export function getClaims(): ExampleClaims { return auth.getJWTClaims() } ``` # Changelog The latest changes and improvements in Modus Welcome to the Modus changelog! Here you'll find the latest improvements to the Modus framework. Stay informed about what's new and what's changed to make the most out of Modus. Here's a short summary of the larger items shipped with each major or minor release. For a more detailed list of changes, please refer to [the full change log in GitHub](https://github.com/hypermodeinc/modus/blob/main/CHANGELOG.md). ## Version history | Version | Date | Description | | ------- | ---------- | ------------------------------------------------------------ | | 0.17.x | 2025-01-24 | MySQL support, local model tracing, and OpenAI improvements. | | 0.16.x | 2024-12-23 | Local time and time zone support | | 0.15.x | 2024-12-13 | Neo4j support | | 0.14.x | 2024-11-23 | Modus API Explorer + in-code documentation | | 0.13.x | 2024-10-17 | First release of Modus as an open source framework 🎉 | Stay tuned for more updates and improvements as we continue to enhance Modus! # Data Fetching Pull data into your app Modus makes it simple to fetch data from external sources. The specific data source you're retrieving from determines the method you use to interact with it. ## Fetching from databases ### PostgreSQL PostgreSQL is a powerful, open source relational database system. Modus provides a simple way to interact with PostgreSQL databases with the `postgresql` APIs. Here is an example of fetching a person from a PostgreSQL database using the Modus SDK: ```go Go package main import ( "github.com/hypermodeinc/modus/sdk/go/pkg/postgresql" ) // the name of the PostgreSQL connection, as specified in the modus.json manifest const connection = "my-database" type Person struct { Name string `json:"name"` Age int `json:"age"` } func GetPerson(name string) (*Person, error) { const query = "select * from persons where name = $1" rows, _, _ := postgresql.Query[Person](connection, query, name) return &rows[0], nil } ``` ```ts AssemblyScript import { postgresql } from "@hypermode/modus-sdk-as" // the name of the PostgreSQL connection, as specified in the modus.json manifest const connection = "my-database" @json class Person { name!: string age!: i32 } export function getPerson(name: string): Person { const query = "select * from persons where name = $1" const params = new postgresql.Params() params.push(name) const response = postgresql.query(connection, query, params) return response.rows[0] } ``` ### Dgraph Dgraph is a distributed, transactional graph database. Modus offers an easy way to query and mutate data in Dgraph with the `dgraph` APIs. Here is an example of fetching a person from a Dgraph database using the Modus SDK: ```go Go package main import ( "encoding/json" "github.com/hypermodeinc/modus/sdk/go/pkg/dgraph" ) // the name of the Dgraph connection, as specified in the modus.json manifest const connection = "my-dgraph" // declare structures used to parse the JSON document returned by Dgraph query. type Person struct { Name string `json:"name,omitempty"` Age int32 `json:"age,omitempty"` } // Dgraph returns an array of Persons type GetPersonResponse struct { Persons []*Person `json:"persons"` } func GetPerson(name string) (*Person, error) { statement := `query getPerson($name: string!) { persons(func: eq(name, $name)) { name age } } ` variables := map[string]string{ "$name": name, } response, _ := dgraph.Execute(connection, &dgraph.Request{ Query: &dgraph.Query{ Query: statement, Variables: variables, }, }) var data GetPersonResponse json.Unmarshal([]byte(response.Json), &data) return data.Persons[0], nil } ``` ```ts AssemblyScript import { dgraph } from "@hypermode/modus-sdk-as" import { JSON } from "json-as" // the name of the Dgraph connection, as specified in the modus.json manifest const connection: string = "my-dgraph" // declare classes used to parse the JSON document returned by Dgraph query. @json class Person { name: string = "" age: i32 = 0 } // Dgraph returns an array of objects @json class GetPersonResponse { persons: Person[] = [] } export function getPerson(name: string): Person { const statement = ` query getPerson($name: string) { persons(func: eq(name, $name)) { name age } }` const vars = new dgraph.Variables() vars.set("$name", name) const resp = dgraph.execute( connection, new dgraph.Request(new dgraph.Query(statement, vars)), ) const persons = JSON.parse(resp.Json).persons return persons[0] } ``` ### Neo4j Neo4j is a graph database management system. Modus provides a simple way to query and mutate data in Neo4j with the `neo4j` APIs. Here is an example of mutating & fetching a person from a Neo4j database using the Modus SDK: ```go Go package main import ( "github.com/hypermodeinc/modus/sdk/go/pkg/neo4j" ) const host = "my-database" func CreatePeopleAndRelationships() (string, error) { people := []map[string]any{ {"name": "Alice", "age": 42, "friends": []string{"Bob", "Peter", "Anna"}}, {"name": "Bob", "age": 19}, {"name": "Peter", "age": 50}, {"name": "Anna", "age": 30}, } for _, person := range people { _, err := neo4j.ExecuteQuery(host, "MERGE (p:Person {name: $person.name, age: $person.age})", map[string]any{"person": person}) if err != nil { return "", err } } for _, person := range people { if person["friends"] != "" { _, err := neo4j.ExecuteQuery(host, ` MATCH (p:Person {name: $person.name}) UNWIND $person.friends AS friend_name MATCH (friend:Person {name: friend_name}) MERGE (p)-[:KNOWS]->(friend) `, map[string]any{ "person": person, }) if err != nil { return "", err } } } return "People and relationships created successfully", nil } type Person struct { Name string `json:"name"` Age int64 `json:"age"` } func GetAliceFriendsUnder40() ([]Person, error) { response, err := neo4j.ExecuteQuery(host, ` MATCH (p:Person {name: $name})-[:KNOWS]-(friend:Person) WHERE friend.age < $age RETURN friend `, map[string]any{ "name": "Alice", "age": 40, }, neo4j.WithDbName("neo4j"), ) if err != nil { return nil, err } nodeRecords := make([]Person, len(response.Records)) for i, record := range response.Records { node, _ := neo4j.GetRecordValue[neo4j.Node](record, "friend") name, err := neo4j.GetProperty[string](&node, "name") if err != nil { return nil, err } age, err := neo4j.GetProperty[int64](&node, "age") if err != nil { return nil, err } nodeRecords[i] = Person{ Name: name, Age: age, } } return nodeRecords, nil } ``` ```ts AssemblyScript import { neo4j } from "@hypermode/modus-sdk-as" // This host name should match one defined in the modus.json manifest file. const hostName: string = "my-database" @json class Person { name: string age: i32 friends: string[] | null constructor(name: string, age: i32, friends: string[] | null = null) { this.name = name this.age = age this.friends = friends } } export function CreatePeopleAndRelationships(): string { const people: Person[] = [ new Person("Alice", 42, ["Bob", "Peter", "Anna"]), new Person("Bob", 19), new Person("Peter", 50), new Person("Anna", 30), ] for (let i = 0; i < people.length; i++) { const createPersonQuery = ` MATCH (p:Person {name: $person.name}) UNWIND $person.friends AS friend_name MATCH (friend:Person {name: friend_name}) MERGE (p)-[:KNOWS]->(friend) ` const peopleVars = new neo4j.Variables() peopleVars.set("person", people[i]) const result = neo4j.executeQuery(hostName, createPersonQuery, peopleVars) if (!result) { throw new Error("Error creating person.") } } return "People and relationships created successfully" } export function GetAliceFriendsUnder40(): Person[] { const vars = new neo4j.Variables() vars.set("name", "Alice") vars.set("age", 40) const query = ` MATCH (p:Person {name: $name})-[:KNOWS]-(friend:Person) WHERE friend.age < $age RETURN friend ` const result = neo4j.executeQuery(hostName, query, vars) if (!result) { throw new Error("Error getting friends.") } const personNodes: Person[] = [] for (let i = 0; i < result.Records.length; i++) { const record = result.Records[i] const node = record.getValue("friend") const person = new Person( node.getProperty("name"), node.getProperty("age"), ) personNodes.push(person) } return personNodes } ``` ## Fetching from APIs ### HTTP HTTP protocols underpin RESTful APIs with OpenAPI schemas. Modus provides a convenient way to interact with any external HTTP API using the `http` APIs in the Modus SDK. Here is an example of fetching a person from an HTTP API using the Modus SDK: ```go Go package main import ( "encoding/json" "fmt" "github.com/hypermodeinc/modus/sdk/go/pkg/http" ) // declare structures used to parse the JSON document returned by the REST API type Person struct { Name string `json:"name,omitempty"` Age int32 `json:"age,omitempty"` } func GetPerson(name string) (*Person, error) { url := fmt.Sprintf("https://example.com/api/person?name=%s", name) response, _ := http.Fetch(url) // The API returns Person object as JSON var person Person response.JSON(&person) return person, nil } ``` ```ts AssemblyScript import { http } from "@hypermode/modus-sdk-as" @json class Person { name: string = "" age: i32 = 0 } export function getPerson(name: string): Person { const url = `https://example.com/api/people?name=${name}` const response = http.fetch(url) // the API returns Person object as JSON return response.json() } ``` ### GraphQL GraphQL is a data-centric query language for APIs that allows you to fetch only the data you need. With the `graphql` APIs in the Modus SDK, you can easily fetch data from any GraphQL endpoint. Here is an example of fetching a person from a GraphQL API using the Modus SDK: ```go Go import ( "encoding/json" "github.com/hypermodeinc/modus/sdk/go/pkg/graphql" ) // the name of the GraphQL connection, as specified in the modus.json manifest const connection = "my-graphql-api" // declare structures used to parse the JSON document returned type Person struct { Name string `json:"name,omitempty"` Age int32 `json:"age,omitempty"` } type GetPersonResponse struct { Person *Person `json:"getPerson"` } func GetPerson(name string) (*Person, error) { statement := `query getPerson($name: String!) { getPerson(name: $name) { age name } }` vars := map[string]any{ "name": name, } response, _ := graphql.Execute[GetPersonResponse](connection, statement, vars) return response.Data.Person, nil } ``` ```ts AssemblyScript import { graphql } from "@hypermode/modus-sdk-as" // the name of the GraphQL connection, as specified in the modus.json manifest const connection: string = "my-graphql-api" // declare classes used to parse the JSON document returned @json class Person { name: string = "" age: i32 = 0 } @json class GetPersonResponse { getPerson: Person | null } export function getPerson(name: string): Person | null { const statement = ` query getPerson($name: String!) { getPerson(name: $name) { age name } }` const vars = new graphql.Variables() vars.set("name", name) const response = graphql.execute( connection, statement, vars, ) return response.data!.getPerson } ``` # Using DeepSeek Use the DeepSeek-R1 Model with your Modus app `DeepSeek-R1` is an open source AI reasoning model that rivals the performance of frontier models such as OpenAI's o1 in complex reasoning tasks like math and coding. Benefits of DeepSeek include: * **Performance**: `DeepSeek-R1` achieves comparable results to OpenAI's o1 model on several benchmarks. * **Efficiency**: The model uses significantly fewer parameters and therefore operates at a lower cost relative to competing frontier models. * **Open Source**: The open source license allows both commercial and non-commercial usage of the model weights and associated code. * **Novel training approach**: The research team developed DeepSeek-R1 through a multi-stage approach that combines reinforcement learning, fine-tuning, and data distillation. * **Distilled versions**: The DeepSeek team released smaller, distilled models based on DeepSeek-R1 that offer high reasoning capabilities with fewer parameters. In this guide we review how to use the `DeepSeek-R1` model in your Modus app. ## Options for using DeepSeek with Modus There are two options for using `DeepSeek-R1` in your Modus app: 1. [Use the distilled `DeepSeek-R1` model hosted by Hypermode](#using-the-distilled-deepseek-model-hosted-by-hypermode) Hypermode hosts and makes available the distilled DeepSeek model based on `Llama-3.1-8B` enabling Modus apps to use it in both local development environments and deployed applications. 2. [Use the DeepSeek Platform API with your Modus app](#using-the-deepseek-platform-api-with-modus) Access DeepSeek models hosted on the DeepSeek platform by configuring a DeepSeek connection in your Modus app and using your DeepSeek API key ## Using the distilled DeepSeek model hosted by Hypermode The open source `DeepSeek-R1-Distill-Llama-8B` DeepSeek model is available on Hypermode as a [shared model](/hosted-models#shared-models). This means that we can invoke this model in a Modus app in both a local development environment and also in an app deployed on Hypermode. The `DeepSeek-R1-Distill-Llama-8B` model is a distilled version of the DeepSeek-R1 model which has been fine-tuned using the `Llama-3.1-8B` model as a base model, using samples generated by DeepSeek-R1. Distilled models offer similar high reasoning capabilities with fewer parameters.