From 5a455e4ee71bb8f67b9bace5ec6e1616d2c9183d Mon Sep 17 00:00:00 2001 From: Salakar Date: Mon, 11 Dec 2023 13:18:15 +0000 Subject: [PATCH] refactor: Prepare docs for OSS --- docs.yaml | 64 ++++++++++++++ docs/assets/globe_dark.png | Bin 0 -> 7353 bytes docs/assets/globe_light.png | Bin 0 -> 29643 bytes docs/cli.commands.deploy.mdx | 37 ++++++++ docs/cli.commands.link.mdx | 29 ++++++ docs/cli.commands.login.mdx | 22 +++++ docs/cli.commands.logout.mdx | 15 ++++ docs/cli.commands.unlink.mdx | 17 ++++ docs/cli.mdx | 45 ++++++++++ docs/deployments.build-settings.mdx | 60 +++++++++++++ docs/deployments.domains.mdx | 17 ++++ docs/deployments.environment-variables.mdx | 34 ++++++++ docs/deployments.github-integration.mdx | 22 +++++ docs/deployments.hooks.mdx | 10 +++ docs/deployments.mdx | 51 +++++++++++ docs/deployments.redeployments.mdx | 19 ++++ docs/frameworks.dart-frog.mdx | 27 ++++++ docs/frameworks.jaspr.mdx | 34 ++++++++ docs/frameworks.mdx | 14 +++ docs/getting-started.mdx | 97 +++++++++++++++++++++ docs/index.mdx | 43 +++++++++ docs/infrastructure.cold-starts.mdx | 32 +++++++ docs/infrastructure.cron-jobs.mdx | 58 ++++++++++++ docs/infrastructure.headers.mdx | 27 ++++++ docs/infrastructure.mdx | 87 ++++++++++++++++++ docs/infrastructure.regions.mdx | 90 +++++++++++++++++++ docs/infrastructure.websockets.mdx | 11 +++ 27 files changed, 962 insertions(+) create mode 100644 docs.yaml create mode 100644 docs/assets/globe_dark.png create mode 100644 docs/assets/globe_light.png create mode 100644 docs/cli.commands.deploy.mdx create mode 100644 docs/cli.commands.link.mdx create mode 100644 docs/cli.commands.login.mdx create mode 100644 docs/cli.commands.logout.mdx create mode 100644 docs/cli.commands.unlink.mdx create mode 100644 docs/cli.mdx create mode 100644 docs/deployments.build-settings.mdx create mode 100644 docs/deployments.domains.mdx create mode 100644 docs/deployments.environment-variables.mdx create mode 100644 docs/deployments.github-integration.mdx create mode 100644 docs/deployments.hooks.mdx create mode 100644 docs/deployments.mdx create mode 100644 docs/deployments.redeployments.mdx create mode 100644 docs/frameworks.dart-frog.mdx create mode 100644 docs/frameworks.jaspr.mdx create mode 100644 docs/frameworks.mdx create mode 100644 docs/getting-started.mdx create mode 100644 docs/index.mdx create mode 100644 docs/infrastructure.cold-starts.mdx create mode 100644 docs/infrastructure.cron-jobs.mdx create mode 100644 docs/infrastructure.headers.mdx create mode 100644 docs/infrastructure.mdx create mode 100644 docs/infrastructure.regions.mdx create mode 100644 docs/infrastructure.websockets.mdx diff --git a/docs.yaml b/docs.yaml new file mode 100644 index 00000000..7e5b7b54 --- /dev/null +++ b/docs.yaml @@ -0,0 +1,64 @@ +--- +name: Docs +logo: "/assets/globe_dark.png" +logoDark: "/assets/globe_light.png" +twitter: dart_globe +anchors: +- title: Discord + icon: discord + link: https://invertase.link/globe-discord +theme: "#FFA03F" +favicon: "/assets/globe_dark.png" +sidebar: +- - Overview + - "/" +- - Getting Started + - "/getting-started" +- - CLI + - - - CLI Overview + - "/cli" + - - Deploy + - "/cli.commands.deploy" + - - Link + - "/cli.commands.link" + - - Login + - "/cli.commands.login" + - - Logout + - "/cli.commands.logout" + - - Unlink + - "/cli.commands.unlink" +- - Deployments + - - - Overview + - "/deployments" + - - Build Settings + - "/deployments.build-settings" + - - Domains + - "/deployments.domains" + - - Environment Variables + - "/deployments.environment-variables" + - - GitHub Integration + - "/deployments.github-integration" + - - Hooks + - "/deployments.hooks" + - - Redeployments + - "/deployments.redeployments" +- - Frameworks + - - - Overview + - "/frameworks" + - - Dart Frog + - "/frameworks.dart-frog" + - - Jaspr + - "/frameworks.jaspr" +- - Infrastructure + - - - Overview + - "/infrastructure" + - - Cold Starts + - "/infrastructure.cold-starts" + - - Cron Jobs + - "/infrastructure.cron-jobs" + - - Headers + - "/infrastructure.headers" + - - Regions + - "/infrastructure.regions" + - - WebSockets + - "/infrastructure.websockets" \ No newline at end of file diff --git a/docs/assets/globe_dark.png b/docs/assets/globe_dark.png new file mode 100644 index 0000000000000000000000000000000000000000..f86a4b777c91ea4556384d60731bfe2a5d63149c GIT binary patch literal 7353 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 SZOldh0000`: + +- `--prod` - Creates a deployment which will be promoted to production once built. diff --git a/docs/cli.commands.link.mdx b/docs/cli.commands.link.mdx new file mode 100644 index 00000000..945f9d65 --- /dev/null +++ b/docs/cli.commands.link.mdx @@ -0,0 +1,29 @@ +--- +title: Link a project to Globe +--- + +# Link + +The `link` command links a local Dart project to a Globe project. Linking is required to deploy a project to Globe. + +Project linking is carried out automatically if the project is not yet linked when running the [`deploy`](/docs/cli/commands/deploy) command. + +## Usage + +```bash +globe link +``` + +The `link` command will first prompt, asking whether you wish to continue linking the local project. If you continue, you will be further prompted to: + +1. Select an account you wish to link this project to. +2. Select, or create a new project to link this project to. + +Upon completion, the project will be linked to your Globe account and you will be able to deploy the project. + +## Details + +A linked project stores metadata about the project in the local `.dart_tool/dart_globe` directory on your machine. This metadata is used to identify the project when deploying to Globe. If not already, +the `.dart_tool` directory should not be committed to your Git repository. You can add this directory to your `.gitignore` file to ensure it is not committed. + +If you accidentally delete this directory, or run the [`unlink`](/docs/cli/commands/unlink) command, you will need to re-link the project before you can deploy again. diff --git a/docs/cli.commands.login.mdx b/docs/cli.commands.login.mdx new file mode 100644 index 00000000..1c9213a2 --- /dev/null +++ b/docs/cli.commands.login.mdx @@ -0,0 +1,22 @@ +--- +title: Login to Globe +--- + +# Login + +The `login` command allows you to authenticate your local CLI with your Globe account. This is required before you can deploy any projects. + +## Usage + +```bash +globe login +``` + +A new browser window will open prompting you to login to your Globe account. Once you have logged in, you will be redirected to a page notifying you of a successful login. You can close this page once complete. +Globe will store a secure token on your local machine to authenticate future requests. + +If you wish to [logout](/docs/cli/commands/logout), you can run the following command: + +```bash +globe logout +``` diff --git a/docs/cli.commands.logout.mdx b/docs/cli.commands.logout.mdx new file mode 100644 index 00000000..452bd1f4 --- /dev/null +++ b/docs/cli.commands.logout.mdx @@ -0,0 +1,15 @@ +--- +title: Logout of Globe +--- + +# Logout + +The `logout` command allows removes any prior authentication tokens from your local machine. Once you have logged out, you will need to login again before you can deploy any projects. + +## Usage + +```bash +globe logout +``` + +A message will be displayed notifying you that you have been logged out. diff --git a/docs/cli.commands.unlink.mdx b/docs/cli.commands.unlink.mdx new file mode 100644 index 00000000..149b2b16 --- /dev/null +++ b/docs/cli.commands.unlink.mdx @@ -0,0 +1,17 @@ +--- +title: Unlink a project from Globe +--- + +# Unlink + +The `unlink` command removes any previously linked projects from the current local project. This will prevent you from deploying the project to Globe. + +## Usage + +```bash +globe unlink +``` + +## Details + +The command removes the `.dart_tool/dart_globe` directory from the local project. This directory stores metadata about the project, which is used to identify the project when deploying to Globe. diff --git a/docs/cli.mdx b/docs/cli.mdx new file mode 100644 index 00000000..05a84444 --- /dev/null +++ b/docs/cli.mdx @@ -0,0 +1,45 @@ +--- +title: Globe CLI +description: Install and start deploying using the Globe CLI. +--- + +# Globe CLI + +The Globe CLI is a command line tool that helps you deploy and interact with your Dart applications via Globe. + +## Pre-requisites + +To get started, you must have the [Dart SDK](https://dart.dev/get-dart) installed on your machine. + +## Installation + +To install the Globe CLI, run the following command: + +```bash +dart pub global activate globe_cli +``` + +Once installed, you can access the `globe` executable from anywhere on your machine. + +To login with your Globe account, run the following command: + +```bash +globe login +``` + +## Commands + +The Globe CLI provides the following commands, which can be accessed by running `globe `: + +- [`deploy`](/cli/commands/deploy) - Deploy your Dart application to Globe directly from the command line. +- [`link`](/cli/commands/link) - Link a local project to a Globe project. +- [`login`](/cli/commands/login) - Login to your Globe account. +- [`logout`](/cli/commands/logout) - Logout of your Globe account. +- [`unlink`](/cli/commands/unlink) - Unlink a local project from a Globe project. + +## Global Flags + +All CLI commands have global flags that can be used to modify the behavior of the command, which can be accessed by running `globe --`. + +- `--help` - Display help information for the command. +- `--verbose` - Display verbose output for the command, such as API calls and debug information. diff --git a/docs/deployments.build-settings.mdx b/docs/deployments.build-settings.mdx new file mode 100644 index 00000000..7e9bab14 --- /dev/null +++ b/docs/deployments.build-settings.mdx @@ -0,0 +1,60 @@ +--- +title: Build Settings +--- + +# Build Settings + +Build settings allow you to configure how your project is build during a new deployment. You can configure your build settings via the dashboard: **Project** -> **Settings** -> **General**. + +![Project Settings](/docs/project-settings.png) + +## Settings + +You can configure the following build settings: + +- [Dart Version](#dart-version) +- [Root Directory](#root-directory) +- [Framework Preset](#framework-preset) +- [Build Command](#build-command) +- [Entrypoint](#entrypoint) + +--- + +### Dart Version + +The Dart version to use when building your project. By default, Globe will use the latest stable version of Dart. Currently Dart supports `stable` and `beta` version branches. + +Support for specific tagged versions is coming soon. + +--- + +### Root Directory + +The root directory is the directory which contains your project source. If working with mono-repositories of a directory structure where your application is not at the root where the `pubspec.yaml` file exists. + +--- + +### Framework Preset + +Globe supports existing [frameworks](/docs/frameworks). If a framework is specified here, Globe will use default values required to build and deploy your framework application. If you'd like to override these values, you can do so by specifying a value in the specific build settings. + +--- + +### Build Command + +A build command can be executed after project dependencies have been installed. This is useful if you need to run a build step before your application can be deployed. + +If using a **Framework Preset**, a default value will be used instead. You can override this by specifying a value here. + +--- + +### Entrypoint + +When building your application, the build system requires an entrypoint file which is used to bootstrap your application. The is typically the `lib/main.dart` file in your project +which contains a `main` function. + +This setting can be configured if your application uses a different entrypoint file. + +If using a **Framework Preset**, a default value will be used instead. You can override this by specifying a value here. + +--- diff --git a/docs/deployments.domains.mdx b/docs/deployments.domains.mdx new file mode 100644 index 00000000..fd69f7f2 --- /dev/null +++ b/docs/deployments.domains.mdx @@ -0,0 +1,17 @@ +--- +title: Domains +--- + +# Domains + +A domain is the address of your deployment. + +Each deployment, production or preview gets assigned a unique domain which directly points to the specific deployment. If the deployment is not yet ready or successful, visiting a domain will redirect you to the dashboard for that deployment. + +## Production domains + +Each project gets assigned a unique domain when it is created. Whenever you successfully deploy to the production environment, the production domain will be updated to point to the latest deployment. + +### Custom Domains + +Custom domains are not yet supported. We are working on it. diff --git a/docs/deployments.environment-variables.mdx b/docs/deployments.environment-variables.mdx new file mode 100644 index 00000000..435fab37 --- /dev/null +++ b/docs/deployments.environment-variables.mdx @@ -0,0 +1,34 @@ +--- +title: Environment Variables +--- + +# Environment Variables + +Environment variables are a set of named values that can affect the way running processes will behave on a deployment. An environment variable can be used to store secret values which are not safe to be stored in the codebase. You may wish to use +different values for different environments, such as development, preview, and production to connect to different databases or API endpoints. + +## Creating environment variables + +To create a new environment variable, navigate to the **Environment Variables** tab in the **Settings** section of the project on the dashboard. Click the **Add Variable** button. + +![New Environment Variable](/docs/new-env-variable.png) + +Enter the name and value of your variable. All values are encrypted, and can only be viewed by a specifically viewing it on the dashboard. + +It is also possible to specify what environments the variable should be available in. + +## Accessing environment variables + +Within your Dart code, you can access environment variables using the `Platform.environment` map. + +```dart +String? value = Platform.environment['MY_VARIABLE']; +``` + +## System environment variables + +The following environment variables are available on all deployments: + +| Name | Description | +| ------ | ----------------------------------------------- | +| `PORT` | The port that the application should listen on. | diff --git a/docs/deployments.github-integration.mdx b/docs/deployments.github-integration.mdx new file mode 100644 index 00000000..4e506e4f --- /dev/null +++ b/docs/deployments.github-integration.mdx @@ -0,0 +1,22 @@ +--- +title: GitHub Integration +--- + +# GitHub Integration + +Globe integrates with GitHub via the [GitHub Application](https://github.com/apps/dart-globe). + +## Features + +- Import a new project from GitHub, specifying the repository, root directory and branch of your Dart project. +- Create new deployments when you push to GitHub. +- Trigger production builds whenever you push to a specified production branch. +- Show deployment status on each commit. + +## Authorization + +TODO: Add authorization instructions. + +## Configuration + +TODO: Add configuration instructions. diff --git a/docs/deployments.hooks.mdx b/docs/deployments.hooks.mdx new file mode 100644 index 00000000..4bfd01dc --- /dev/null +++ b/docs/deployments.hooks.mdx @@ -0,0 +1,10 @@ +--- +title: Deployment Hooks +--- + +# Deployment Hooks + +Deployment hooks allow you to provide a webhook endpoint which will be called throughout the lifecycle of a deployment, for example +when a deployment is created, built, or deployed. + +Deployment hooks are an in development feature and will be available soon. diff --git a/docs/deployments.mdx b/docs/deployments.mdx new file mode 100644 index 00000000..d1b119a8 --- /dev/null +++ b/docs/deployments.mdx @@ -0,0 +1,51 @@ +--- +title: Deploying to Globe +--- + +# Deployments + +A deployment can be triggered when you wish to deploy new code to your Globe project. In Globe, a deployment can be staged to two environments; **Production** or **Preview**. +When a Production deployment has successfully built, your domains will be updated automatically to point to the new deployment. Preview deployments on the other are accessible +via a URL unique to the deployment. This allows you to test and share your changes on a live environment before deploying to production. + +![Deployments](/docs/deployments.png) + +Once deployments have successfully built, you can access the unique URL from the Globe dashboard. + +## Creating a preview deployment + +To create a new preview deployment, you have two options: + +### via the CLI + +Run the [`deploy`](/docs/cli/commands/deploy) command. By default, deployments from the CLI are staged as a preview deployment: + +```bash +globe deploy +``` + +### via GitHub + +Using the [GitHub Integration](/docs/deployments/github-integration), you can trigger a preview deployment by pushing to a branch which +is not defined as the production branch in your projects settings. + +To learn more, view the [GitHub Integration](/docs/deployments/github-integration) documentation. + +## Creating a production deployment + +To create a new production deployment, you have two options: + +### via the CLI + +Run the [`deploy`](/docs/cli/commands/deploy) command with the `--prod` flag: + +```bash +globe deploy --prod +``` + +### via GitHub + +Using the [GitHub Integration](/docs/deployments/github-integration), you can trigger a production deployment by pushing to the branch +specified in your projects settings. By default, this is the `main` branch. + +To learn more, view the [GitHub Integration](/docs/deployments/github-integration) documentation. diff --git a/docs/deployments.redeployments.mdx b/docs/deployments.redeployments.mdx new file mode 100644 index 00000000..62d48027 --- /dev/null +++ b/docs/deployments.redeployments.mdx @@ -0,0 +1,19 @@ +--- +title: Redeployments +--- + +# Redeployments + +Redeployments are the process of triggering a new deployment build from a previous deployment. This is useful if you update build settings or environment variables and want to apply those changes without making any other changes to your code. + +Redeployments apply any new build settings and environment variables you might have updated since the original deployment. + +## Redeploying from the Dashboard + +To redeploy from the dashboard, click the **Redeploy** button on the deployment you want to redeploy: + +![Redeployments](/docs/redeploy.png) + +## Redeploying other redeployments + +When redeploying a redeployment, the original deployment will be used as the base for the new deployment. diff --git a/docs/frameworks.dart-frog.mdx b/docs/frameworks.dart-frog.mdx new file mode 100644 index 00000000..381db969 --- /dev/null +++ b/docs/frameworks.dart-frog.mdx @@ -0,0 +1,27 @@ +--- +title: Deploying a Dart Frog application +--- + +# Deploying a Dart Frog application + +[Dart Frog](https://dartfrog.vgv.dev/) is a fast, minimalistic backend framework for Dart which is built on top of [Shelf](https://pub.dev/packages/shelf). Dart Frog supports routing, middleware, and DI out of the box with a local CLI to assist with development. + +Follow their [documentation](https://dartfrog.vgv.dev/docs/overview) for assistance with installing the CLI. + +## Create a new project + +Create a new Dart Frog project by running the following command: + +```bash +dart_frog create my_dart_frog_app +``` + +Start a local development server with the following command: + +```bash +dart_frog dev +``` + +## Deploying to Globe + +Deploying your project through the CLI or the dashboard will work as normal. Globe will detect Dart Frog is being used and automatically select it as a framework preset on your project settings. No additional steps are required. diff --git a/docs/frameworks.jaspr.mdx b/docs/frameworks.jaspr.mdx new file mode 100644 index 00000000..a41c62a2 --- /dev/null +++ b/docs/frameworks.jaspr.mdx @@ -0,0 +1,34 @@ +--- +title: Deploying a Jaspr application +--- + +# Deploying a Dart Frog application + +[Jaspr](https://pub.dev/packages/jaspr) is a modern web framework for building websites in Dart with support for both client-side and server-side rendering. + +Follow the [documentation](https://pub.dev/packages/jaspr#get-started) for assistance with installing the Jaspr CLI. + +## Create a new project + +Create a new Dart Frog project by running the following command: + +```bash +jaspr create my_jaspr_app +``` + +Start a local development server with the following command: + +```bash +dart run jaspr serve +``` + +## Deploying to Globe + +Before deploying your application to Globe, ensure you have the Server Side Rendering disabled flag disabled in your `pubspec.yaml` file: + +```yaml +jaspr: + uses-ssr: false +``` + +This will enable Globe to build and serve your application instead. Globe will detect Jaspr is being used and automatically select it as a framework preset on your project settings. No additional steps are required. diff --git a/docs/frameworks.mdx b/docs/frameworks.mdx new file mode 100644 index 00000000..301ccc1b --- /dev/null +++ b/docs/frameworks.mdx @@ -0,0 +1,14 @@ +--- +title: Frameworks +--- + +# Frameworks + +Globe aims to be framework agnostic, however, some frameworks have additional steps like build commands that need to be ran before deployment. Globe supports these frameworks by automatically providing default build settings for each framework that can used to build and deploy your application. + +## Supported Frameworks + +On-going work is being done to support more frameworks. If you'd like to see a framework supported please submit a [feature request](https://invertase.canny.io/globe). + +- [Dart Frog](/docs/frameworks/dart-frog) +- [Jaspr](/docs/frameworks/jaspr) diff --git a/docs/getting-started.mdx b/docs/getting-started.mdx new file mode 100644 index 00000000..c1497b9b --- /dev/null +++ b/docs/getting-started.mdx @@ -0,0 +1,97 @@ +--- +title: Getting started with Globe +--- + +# Getting Started + +To get started using Globe, ensure you have [logged in](/login) to your account and have access. Globe is current in private alpha, so only select users have access. + +## Install the Globe CLI (optional) + +The Globe CLI is a command line tool that allows you to interact with your Globe account and deploy from the command line. It is not required to use Globe, but it can be useful for some workflows. + +To install the Globe CLI, run the following command: + +```bash +dart pub global activate globe_cli +``` + +Once installed, you can access the globe executable from anywhere on your machine. + +To login with your Globe account, run the following command: + +```bash +globe login +``` + +To learn more about the CLI, view the [documentation](/docs/cli). + +## Create a new Dart project + +To get started with, we're going to create a simple Dart application which will be deployed to Globe. We'll use [Shelf](https://pub.dev/packages/shelf) to start a HTTP server which can handle requests and log some information about our application. Let's go ahead and create a basic Dart application. In your terminal, run the following command where you'd like to create a project: + +```bash +dart create my_shelf_app +``` + +Rename the file `lib/my_shelf_app.dart` to `lib/main.dart` - this is the entrypoint for our application. Next, install the [`shelf`](https://pub.dev/packages/shelf) package via pub: + +```bash +dart pub add shelf +``` + +Copy the following code into `lib/main.dart`: + +```dart +import 'dart:io'; + +import 'package:shelf/shelf.dart'; +import 'package:shelf/shelf_io.dart' as shelf_io; + +void main() async { + final handler = + const Pipeline().addMiddleware(logRequests()).addHandler(_echoRequest); + + final server = await shelf_io.serve( + handler, + InternetAddress.anyIPv4, + int.tryParse(Platform.environment['PORT'] ?? '8080') ?? 8080, + ); + + print('Serving at http://${server.address.host}:${server.port}'); +} + +Response _echoRequest(Request request) => + Response.ok('Request for "${request.url}"... worked!'); +``` + +To test your application out locally, run `dart run lib/main.dart` and visit `http://0.0.0.0:8080/hello` in your browser. You should see the following: `Request for "hello"... worked!`. + +This is a basic HTTP server that will respond to any request with a 200 OK response and the body `Request for "${request.url}"... worked!`, whilst also logging out information about the request. From here you could build out a full application yourself, adding in routing, database calls, middleware and more. For this example, we'll keep it simple. + +## Deploying to Globe + +Now that we have a basic application, we can deploy it to Globe. Deploying to Globe is simple; run the `globe deploy` command from your project root in the terminal: + +```bash +globe deploy +``` + +The first time you deploy, you'll: + +1. Be prompted to continue with setup of the deployment (press `Y`) +1. Enter a name for your project: Enter: `my-shelf-app` + +After waiting for a couple of seconds, you'll be shown that your new deployment has been queued and be provided a unique URL for that deployment. You can visit this URL in your browser to view the build logs and deployment status. + +## Viewing your deployment logs + +Once complete, your deployment will be available via the URL shown in the dashboard. Each deployment has it's own unique URL, with a `globeapp.dev` domain. Click the URL and you'll be shown the `Request for "hello"... worked!` message which your saw when running your application locally. + +Since our application has both a `print` statement and and a middleware logger (provided by Shelf), we're able to inspect those logs in realtime from the dashboard. Click the "Logs" tab in the dashboard to view the logs for your deployment. Each time your refresh the deployment URL, a new log will appear in the dashboard! Any errors (which emit to `stderr`) will be highlighted in red. + +## Further reading + +- [Learn about to manage your deployments](/docs/deployments) +- [Learn about the Globe CLI](/docs/cli) +- [Integrate with GitHub](/docs/deployments/github-integration) diff --git a/docs/index.mdx b/docs/index.mdx new file mode 100644 index 00000000..7ecdb4a9 --- /dev/null +++ b/docs/index.mdx @@ -0,0 +1,43 @@ +--- +title: Getting started with Globe +--- + + + +Globe is a deployment platform for [Dart](https://dart.dev/) developers, that allows you to deploy your Dart applications to a globally +distributed service without the need to manage servers, networks, or scaling. The core principle of Globe is a simplified deployment service +with great developer experience in mind. + +## Features + +Some of the core features of Globe include: + +- **Deployments**: Deploy your Dart applications to Globe with a single command or automatically on every push to your Git repository. Globe supports both previewing and production deployments. +- **GitHub Integration**: Globe integrates with your existing CI/CD pipeline to automatically deploy your applications on every push to your Git repository. +- **Custom domains**: Assign custom domains to your projects, with SSL certificates automatically generated and renewed for you. +- **Organization support**: Globe supports organizations, allowing you to manage multiple projects and users under a single account. + +## Why Globe? + +Dart on the server is a great choice for building server based applications, such as APIs, websites, and more. The language has great support for building these applications with packages such +as [Shelf](https://pub.dev/packages/shelf) and [Dart Frog](https://pub.dev/packages/dart_frog) providing a great base for getting started. However, deploying these applications can be a challenge for +a number of reasons: + +- Developers require experience with cloud providers such as Google Cloud Platform (GCP), Amazon Web Services (AWS), Digital Ocean and more. +- Familiarity with tools such as Docker, Kubernetes, and Terraform is required to deploy applications. +- Critical components such as scaling, load balancing, and SSL certificates must be configured and monitored. +- Integrating with CI/CD flows can be challenging, especially when deploying multiple environments and preview deployments. +- This architecture comes with challenges such as managing cold starts, region based latency, and more. + +Globe aims to solve these problems by providing a simple, easy to use deployment platform for Dart developers. Globe utilizes a number of cloud providers to provide a globally distributed service with the following benefits: + +- Zero management overhead of servers, networking and scaling. +- Globally available regions. Globe deployments of your application are available around the world, providing low latency to your users and high availability. +- Globe automatically scales your application to meet demand. +- Built-in support for preview deployments, allowing you to preview every change, commit or PR before deploying to production. + +Simply run the `deploy` command or push to your Git repository and Globe will create a globally managed deployment for you within seconds. diff --git a/docs/infrastructure.cold-starts.mdx b/docs/infrastructure.cold-starts.mdx new file mode 100644 index 00000000..94f7e088 --- /dev/null +++ b/docs/infrastructure.cold-starts.mdx @@ -0,0 +1,32 @@ +--- +title: Cold Starts +--- + +# Cold Starts + +Globe infrastructure is built in a way to minimize and reduce the impact of cold starts. When a request starts a new container on demand this is known as a cold start. + +In Globe however we have 3 levels of 'starts'; + +- `cold` - Container image has to be fetched, cached and started. + - This is usually when your deployment has not yet had any requests in a specific compute region or has been idle for an extended period of time. +- `warm` - Container image is already fetched and cached but not started. + - This happens when a `hot` running container becomes idle for a period of time in a specific region and the running container is paused. +- `hot` - Container image is already fetched, cached and still running. + - This happens when requests in a specific region have kept the container from idling. + +Overall you should experience a reduced number of cold starts compared to other cloud providers as well as a reduced cold start time. During the private preview you can expect to see some cold starts of around 500ms. We're still working on reducing this number. + +## Mitigating cold starts + +Whilst it may not necessary for every application; there are a number of ways you can help mitigate the impact of cold starts specific to your deployment; + +- Reduce the size of your application by removing unnecessary dependencies and imports. + +## Detecting cold starts + +Each request to a Globe deployment has a custom header appended to it. The `x-globe-temperature` header will have a value of the start state: + +- `cold` +- `warm` +- `hot` diff --git a/docs/infrastructure.cron-jobs.mdx b/docs/infrastructure.cron-jobs.mdx new file mode 100644 index 00000000..05e27fad --- /dev/null +++ b/docs/infrastructure.cron-jobs.mdx @@ -0,0 +1,58 @@ +--- +title: Cron Jobs +--- + +# Cron Jobs + +Cron jobs can be used to periodically perform a HTTP request to your production Globe deployment. You can specify the cron schedule and the path to request. + +Cron jobs only work with production deployments. If your project does not have a production deployment, the cron execution will be ignored and no events will be shown. + +## Creating a Cron Job + +Cron jobs can be created via the `globe.yaml` file in the root of your application project, for example: + +```yaml +crons: + - id: refresh_auth_token + schedule: '0,45 * * * *' + path: '/auth/refresh' +``` + +In the above example, the cron job will be executed every 45 minutes. + +Whenever a new production deployment is successful, the cron jobs will be sychronized and shown in the dashboard. + +## Handling Cron Jobs + +Your application code should handle the cron job request via the path specified (`/auth/refresh` in the above example). The request will be a `POST` request. + +For example, using [`shelf_router`]() this can be achived as follows: + +```dart +import 'package:shelf_router/shelf_router.dart'; + +var app = Router(); + +app.post('/auth/refresh', (Request request) async { + // Some logic to refresh the auth token + return Response.ok('ok'); +}); +``` + +If a Cron Job returns a response with a status code other than one in the `2xx` range, this will show on the dashboard as an error. + +## Cron Job IDs + +The `id` field in the list represents the unique ID of the cron job. This is used to identify the cron job on the dashboard, and store event logs. +If a specific ID is removed, and later added, new event logs will be stored alongside any previous logs. Also note: + +- The list cannot contain duplicate IDs. +- The maximum length of an ID is 50 characters. +- The ID can only contain lowecase letters, numbers, and underscores (`a-z0-9_`). + +## Timezone + +At present, all cron jobs are executed in the UTC (Universal Coordinated Time) timezone, so you must take this into account when specifying the schedule. + +Please [upvote this feature request](https://invertase.canny.io/globe/p/custom-cron-job-timezones) if you would like support for custom timezones. diff --git a/docs/infrastructure.headers.mdx b/docs/infrastructure.headers.mdx new file mode 100644 index 00000000..a2060d99 --- /dev/null +++ b/docs/infrastructure.headers.mdx @@ -0,0 +1,27 @@ +--- +title: Headers +--- + +# Headers + +Each request to a deployment on Globe is automatically provided with a number of request headers which you could use in your application to customize how your application responds to the request. + +| **Header** | **Type** | **Description** | +| ---------------------- | :------: | --------------------------------------------------------------------------------------------------------- | +| `x-globe-request-id` | String | The unique ID for this request. | +| `x-globe-dc-id` | String? | The IATA code of where this request is executing. | +| `x-globe-dc-eu` | String? | A "true" or "false" string value of whether this request is executing within a data center within the EU. | +| `x-globe-dc-city` | String? | The data center city name of where this request is executing. | +| `x-globe-dc-country` | String | The data center ISO-3166-1 alpha-2 code of this request (e.g. "US"). | +| `x-globe-dc-continent` | String | The data center continent (e.g. "Europe"). | +| `x-globe-dc-latitude` | String | The data center location latitude. | +| `x-globe-dc-longitude` | String | The data center location longitude. | +| `x-globe-eu` | String? | A "true" or "false" string value of whether this request originated from within the EU. | +| `x-globe-latitude` | String? | The location latitude of where this request originated from. | +| `x-globe-longitude` | String? | The location longitude of where this request originated from. | +| `x-globe-city` | String? | The closest city of where this request originated from. | +| `x-globe-country` | String? | The ISO-3166-1 alpha-2 code of where this request originated from (e.g. "US"). | +| `x-globe-region` | String? | The location region of the incoming request (e.g. "Texas"). | +| `x-globe-region-code` | String? | The region code of the incoming request (as a ISO-3166-2 code e.g. "TX"). | +| `x-globe-continent` | String? | The continent of the incoming request (e.g. "Europe"). | +| `x-globe-tz` | String? | The location timezone of the incoming request, e.g. "America/Chicago". | diff --git a/docs/infrastructure.mdx b/docs/infrastructure.mdx new file mode 100644 index 00000000..5a9e0981 --- /dev/null +++ b/docs/infrastructure.mdx @@ -0,0 +1,87 @@ +--- +title: Infrastructure +--- + +# Infrastructure + +Globe is built on top of multiple cloud compute providers to provide a scalable and reliable service. The following sections provide an overview of the infrastructure used to run Globe & how you can integrate your Dart applications. + +## How it works + +Each time you create a new successful deployment, the Globe deployment service creates an executable of your Dart project. This executable is distributed to our global private deployment registry. + +When a user makes a request to your application, they first end up connecting to our global edge network (300+ locations). Our edge network provides zero latency routing management and allows us to route your user to the nearest [Globe compute region](/docs/infrastructure/regions) where our infrastructure will start a container in isolation on demand if necessary. + +Our edge network routing service is also responsible for features such as DDoS mitigation, instant rollbacks, A/B testing deployments and more. + +## Containers + +Globe uses container technology to start your Dart applications on demand. The platform will automatically scale your application based on the number of requests it receives. +Containers are created on demand in the closest [region](/docs/infrastructure/regions) to where a user request originates from. + +An on demand container environment means you should: + +- Not rely on having a persistent file system. +- Not rely on having persistent memory. + +Even if a user is performing multiple requests from the same region, it is not guaranteed that the same container will be used to handle the request. Treat each request as though it is a new environment. + +## Serverless + +Globe currently does not yet have a functions / serverless model, however we are experimenting providing this via upcoming technologies like WASM. Please consider providing your thoughts and your upvote on [this feature request](https://invertase.canny.io/globe/p/faas-serverless) if this is something that interests you. + +## Starting a server + +The Globe infrastructure expects that your Dart application starts listening to a port whenever executed. During the build phase of a deployment, +a check will take place to ensure that your application is listening to a port. If this is not the case, the deployment will fail. + +Using a package such as [shelf](https://pub.dev/packages/shelf), you can easily start a server which accepts in-bound HTTP requests by accessing +the `PORT` [environment variable](/docs/deployments/environment-variables): + +```dart +import 'dart:io'; + +import 'package:shelf/shelf.dart'; +import 'package:shelf/shelf_io.dart' as shelf_io; + +void main() async { + final handler = + const Pipeline().addHandler((Request request) { + return Response.ok('Hello, World!'); + }); + + final server = await shelf_io.serve( + handler, + // Use any IPv4 address. + InternetAddress.anyIPv4, + // Use the PORT environment variable. + int.tryParse(Platform.environment['PORT'] ?? '8080') ?? 8080, + ); +} +``` + +## Limitations + +Globe imposes a few request limits to ensure that the platform is not abused. + +#### File System + +You should only rely on the file system for temporary storage such as processing of files. Any files created during a request will be removed once the request is finished. + +#### Execution Timeout + +A single request is allowed to run for a maximum duration. If a request takes longer than the allowed duration, the request will be terminated. + +During private preview the maximum duration is currently **5 seconds**. We may increase this based on feedback and also allow higher limits for paid plans in future. + +#### CPU Usage + +You cannot perform CPU heavy tasks for an extended period of time (determined by the current infrastructure load). Any requests which exceed the allotted CPU usage will be terminated. + +#### Memory Usage + +Each container has a limited amount of memory available (128MB). Performing on-going memory heavy tasks may result in the request being terminated. We may increase this based on feedback and also allow higher limits for paid plans in future. + +#### FFI + +Usage of FFI (via `dart:ffi`) is not currently supported. We are working on providing this in future. diff --git a/docs/infrastructure.regions.mdx b/docs/infrastructure.regions.mdx new file mode 100644 index 00000000..2cb93975 --- /dev/null +++ b/docs/infrastructure.regions.mdx @@ -0,0 +1,90 @@ +--- +title: Regions +--- + +# Regions + +The Globe edge network is located in some [300 cities in 100+ countries](https://www.cloudflare.com/en-gb/network/) and is powered by Cloudflare’s network - which is milliseconds away from virtually every Internet user on the globe. + +Our edge network is responsible for features such as DDoS mitigation, zero latency routing to our compute regions, instant rollbacks, A/B testing deployments and more. + +## Compute Regions + +Globe runs your deployments in 37 (currently) compute regions. + +
+ Compute Regions + +| **Region** | **Country** | **Continent** | +| ----------------------- | -------------- | ------------- | +| MAD (Madrid) | Spain | Europe | +| BRU (Belgium) | Belgium | Europe | +| AMS (Netherlands) | Netherlands | Europe | +| MXP (Milan) | Italy | Europe | +| CDG (Paris) | France | Europe | +| TPE (Taiwan) | Taiwan | Asia | +| NRT (Tokyo) | Japan | Asia | +| KIX (Osaka) | Japan | Asia | +| HEL (Finland) | Finland | Europe | +| TLV (Tel Aviv) | Israel | Asia | +| DSM (Iowa) | United States | North America | +| CHS (South Carolina) | United States | North America | +| IAD (Northern Virginia) | United States | North America | +| CMH (Columbus) | United States | North America | +| DFW (Dallas) | United States | North America | +| PDX (Oregon) | United States | North America | +| HKG (Hong Kong) | Hong Kong | Asia | +| ICN (Seoul) | South Korea | Asia | +| SIN (Singapore) | Singapore | Asia | +| CGK (Jakarta) | Indonesia | Asia | +| BOM (Mumbai) | India | Asia | +| DEL (Delhi) | India | Asia | +| SYD (Sydney) | Australia | Oceania | +| MEL (Melbourne) | Australia | Oceania | +| WAW (Warsaw) | Poland | Europe | +| TRN (Turin) | Italy | Europe | +| LHR (London) | United Kingdom | Europe | +| FRA (Frankfurt) | Germany | Europe | +| ZRH (Zurich) | Switzerland | Europe | +| DOH (Doha) | Qatar | Asia | +| YUL (Montreal) | Canada | North America | +| YYZ (Toronto) | Canada | North America | +| GRU (Sao Paulo) | Brazil | South America | +| SCL (Santiago) | Chile | South America | +| LAX (Los Angeles) | United States | North America | +| SLC (Salt Lake City) | United States | North America | +| LAS (Las Vegas) | United States | North America | + +
+ +Globe compute regions will continue expanding into the following regions: Berlin (Germany), Dammam (Kingdom of Saudi Arabia), Querétaro (Mexico), Malaysia, Thailand, New Zealand, Greece, Norway, South Africa, Austria and Sweden. + +## Preferred Regions + +In some cases, executing a request in a region closest to the user may not be desirable. Consider a user who hits your application from Europe but your application uses a region specific database located in the United States. + +Multiple outbound requests from Europe to the US data center could result in potentially long response times for +the user whilst data is sent across the network. In this scenario, it may make more sense to handle the request in the US data center, and then +return the response to the user from there. + +This can be achieved by specifying preferred region(s). When preferred regions are specified Globe will: + +- if a single preferred region is provided: execute the request in the preferred region + - if the specified region is unhealthy or experiencing an outage, then default global routing behavior is used and the nearest healthy region executes the request. +- if multiple regions are provided, Globe will choose the closest preferred region to the user and execute the request there + - if any of specified regions are unhealthy or experiencing an outage, they will no longer be preferred. + - if all of the specified regions are unhealthy or experiencing an outage, then default global routing behavior is used and the nearest healthy region executes the request. + +### Configure via Project Settings + +Within the Globe dashboard, under **Project** -> **Settings** -> **Preferred Region**, you can select a preferred region for your project. + +When selecting preferred regions from the dashboard, this will only apply to new deployments. + +### Configure via `X-Globe-Preferred-Region` header + +Specify IATA codes from the region list in the `X-Globe-Preferred-Region` header of a request. + +For example: `X-Globe-Preferred-Region: DFW, PDX` + +Note that when specifying a preferred region via the header this will override any preferred region(s) set in the Project Settings dashboard. diff --git a/docs/infrastructure.websockets.mdx b/docs/infrastructure.websockets.mdx new file mode 100644 index 00000000..12e59474 --- /dev/null +++ b/docs/infrastructure.websockets.mdx @@ -0,0 +1,11 @@ +--- +title: Web Sockets +--- + +# Web Sockets + +We Sockets enable you to open a two-way interactive communication session between your clients and server. This is a great way to build real-time applications like chat, games, and monitoring tools. + +Globe has currently disabled Web Sockets connections whilst we monitor our infrastructure during the private preview period. + +If you would like to use this feature, please consider providing your thoughts/use cases and your upvote on this [feature request](https://invertase.canny.io/globe/p/web-socket-support).