Ensure that you have Python 3.10 or later installed on your system. You can check your Python version by running:
python3 --version
If you do not have Python 3.10 or later, you can install it by following the instructions here.
You’ll install all the required Python tools as part of building the application. We’ll cover that in the Build Your Application section below. For now, there’s nothing to do here!
Use the spin new command and the http-rust template to scaffold a new Spin application:
$ spin new
Pick a template to start your application with:
http-c (HTTP request handler using C and the Zig toolchain)
http-csharp (HTTP request handler using C# (EXPERIMENTAL))
http-go (HTTP request handler using (Tiny)Go)
http-grain (HTTP request handler using Grain)
> http-rust (HTTP request handler using Rust)
http-swift (HTTP request handler using SwiftWasm)
http-zig (HTTP request handler using Zig)
redis-go (Redis message handler using (Tiny)Go)
redis-rust (Redis message handler using Rust)
Enter a name for your new application: hello_rust
Project description: My first Rust Spin application
HTTP path: /...
This command created a directory with the necessary files needed to build and run a Rust Spin application. Change to that directory, and look at the files. It looks very much like a normal Rust library project:
$ cd hello_rust
$ tree
.
├── .gitignore
├── Cargo.toml
├── spin.toml
└── src
└── lib.rs
The additional spin.toml file is the manifest file, which tells Spin what events should trigger what components. In this case our trigger is HTTP, for a Web application, and we have only one component, at the route /.... This is a wildcard that matches any route:
This represents a simple Spin HTTP application (triggered by an HTTP request). It has:
A single HTTP trigger, for the /... route, associated with the hello-rust component. /... is a wildcard, meaning it will match any route. When the application gets an HTTP request that matches this route - that is, any HTTP request at all! - Spin will run the hello-rust component.
A single component called hello-rust, whose implementation is in the associated hello_rust.wasm WebAssembly component. When, in response to the HTTP trigger, Spin runs this component, it will execute the HTTP handler in hello_rust.wasm. (We’re about to see the source code for that.)
Now let’s have a look at the code. Below is the complete source
code for a Spin HTTP component written in Rust — a regular Rust function that
takes an HTTP request as a parameter and returns an HTTP response, and it is
annotated with the http_component macro which identifies it as the entry point
for HTTP requests:
use spin_sdk::http::{IntoResponse, Request, Response};
use spin_sdk::http_component;
/// A simple Spin HTTP component.#[http_component]fnhandle_hello_rust(req: Request) -> anyhow::Result<implIntoResponse> {
println!("Handling request to {:?}", req.header("spin-full-url"));
Ok(Response::builder()
.status(200)
.header("content-type", "text/plain")
.body("Hello, Fermyon")
.build())
}
Use the spin new command and the http-ts template to scaffold a new Spin application. (If you prefer JavaScript to TypeScript, the http-js template is very similar.):
$ spin new
Pick a template to start your application with:
http-js (HTTP request handler using Javascript)
> http-ts (HTTP request handler using Typescript)
Enter a name for your new application: hello_typescript
Project description: My first TypeScript Spin application
HTTP path: /...
This command created a directory with the necessary files needed to build and run a TypeScript Spin application. Change to that directory, and look at the files. It looks similar to a lot of NPM projects:
The additional spin.toml file is the manifest file, which tells Spin what events should trigger what components. In this case our trigger is HTTP, for a Web application, and we have only one component, at the route /.... This is a wildcard that matches any route:
spin_manifest_version = 2[application]name = "hello_typescript"version = "0.1.0"authors = ["Your Name <your-name@example.com>"]
description = "My first TypeScript Spin application"[[trigger.http]]route = "/..."component = "hello-typescript"[component.hello-typescript]source = "target/hello-typescript.wasm"exclude_files = ["**/node_modules"]
[component.hello-typescript.build]command = "npm run build"
This represents a simple Spin HTTP application (triggered by an HTTP request). It has:
A single HTTP trigger, for the /... route, associated with the hello-typescript component. /... is a wildcard, meaning it will match any route. When the application gets an HTTP request that matches this route - that is, any HTTP request at all! - Spin will run the hello-typescript component.
A single component called hello-typescript, whose implementation is in the associated hello-typescript.wasm WebAssembly component. When, in response to the HTTP trigger, Spin runs this component, it will execute the HTTP handler in hello-typescript.wasm. (We’re about to see the source code for that.)
Now let’s have a look at the code. Below is the complete source
code for a Spin HTTP component written in TypeScript — a regular function named handleRequest that
takes an HTTP request as a parameter and returns an HTTP response. (The
JavaScript version looks slightly different, but is still a function with
the same signature.) The Spin js2wasm plugin looks for the handleRequest function
by name when building your application into a Wasm module:
Please note: For more information about managing spin templates, see the templates section in the Spin Command Line Interface (CLI) documentation.
This command created a directory with the necessary files needed to build and run a Python Spin application. Change to that directory, and look at the files. It contains a minimal Python application:
$ spin new -t http-py hello-python --accept-defaults
Once the component is created, we can change into the hello-python directory, create and activate a virtual environment and then install the component’s requirements:
$ cd hello-python
Create a virtual environment directory (we are still inside the Spin app directory):
The (venv-dir) will prefix your terminal prompt now:
(venv-dir) user@123-456-7-8 hello-python %
The requirements.txt, by default, contains the references to the spin-sdk and componentize-py packages. These can be installed in your virtual environment using the following command:
The hello-python directory structure created by the Spin http-py template is shown below:
├── app.py
├── spin.toml
└── requirements.txt
The additional spin.toml file is the manifest file, which tells Spin what events should trigger what components. In this case our trigger is HTTP, for a Web application, and we have only one component, at the route /.... This is a wildcard that matches any route.
This represents a simple Spin HTTP application (triggered by an HTTP request). It has:
A single HTTP trigger, for the /... route, associated with the hello-python component. /... is a wildcard, meaning it will match any route. When the application gets an HTTP request that matches this route - that is, any HTTP request at all! - Spin will run the hello-python component.
A single component called hello-python, whose implementation is in the associated app.wasm WebAssembly component. When, in response to the HTTP trigger, Spin runs this component, it will execute the HTTP handler in app.wasm. (We’re about to see the source code for that.)
Now let’s have a look at the code. Below is the complete source
code for a Spin HTTP component written in Python — a regular function named handle_request that
takes an HTTP request as a parameter and returns an HTTP response.
from spin_sdk.http import IncomingHandler, Request, Response
classIncomingHandler(IncomingHandler):
defhandle_request(self, request: Request) -> Response:
return Response(
200,
{"content-type": "text/plain"},
bytes("Hello from the Python SDK!", "utf-8")
)
Use the spin new command and the http-go template to scaffold a new Spin application.
$ spin new
Pick a template to start your application with:
http-c (HTTP request handler using C and the Zig toolchain)
http-empty (HTTP application with no components)
> http-go (HTTP request handler using (Tiny)Go)
http-grain (HTTP request handler using Grain)
http-php (HTTP request handler using PHP)
http-rust (HTTP request handler using Rust)
Enter a name for your new application: hello_go
Description: My first Go Spin application
HTTP base: /
HTTP path: /...
This command created a directory with the necessary files needed to build and run a Go Spin application using the TinyGo compiler. Change to that directory, and look at the files. It looks very much like a normal Go project:
$ cd hello_go
$ tree
.
├── go.mod
├── go.sum
├── main.go
└── spin.toml
The additional spin.toml file is the manifest file, which tells Spin what events should trigger what components. In this case our trigger is HTTP, for a Web application, and we have only one component, at the route /.... This is a wildcard that matches any route.
spin_manifest_version = 2[application]name = "hello_go"version = "0.1.0"authors = ["Your Name <your-name@example.com>"]
description = "My first Go Spin application"[[trigger.http]]route = "/..."component = "hello-go"[component.hello-go]source = "main.wasm"allowed_outbound_hosts = []
[component.hello-go.build]command = "tinygo build -target=wasi -gc=leaking -no-debug -o main.wasm main.go"
This represents a simple Spin HTTP application (triggered by an HTTP request). It has:
A single HTTP trigger, for the /... route, associated with the hello-go component. /... is a wildcard, meaning it will match any route. When the application gets an HTTP request that matches this route - that is, any HTTP request at all! - Spin will run the hello-go component.
A single component called hello-go, whose implementation is in the associated main.wasm WebAssembly component. When, in response to the HTTP trigger, Spin runs this component, it will execute the HTTP handler in main.wasm. (We’re about to see the source code for that.)
Now let’s have a look at the code. Below is the complete source
code for a Spin HTTP component written in Go. Notice where the work is done. The
main function is empty (and Spin never calls it). Instead, the init function
sets up a callback, and passes that callback to spinhttp.Handle to register it as
the handler for HTTP requests. You can learn more about this structure
in the Go language guide.
The Spin template creates starter source code. Now you need to turn that into a Wasm module. The template puts build instructions for each component into the manifest. Use the spin build command to run them:
You can always run this command manually; spin build is a shortcut to save you having to remember it.
As normal for NPM projects, before you build for the first time, you must run npm install:
$ npm install
added 141 packages, and audited 142 packages in 13s
20 packages are looking for funding
run `npm fund` for details
found 0 vulnerabilities
Then run spin build:
$ spin build
Executing the build commandfor component hello-typescript: npm run build
> hello-typescript@1.0.0 build
> npx webpack --mode=production && mkdir -p target && spin js2wasm -o target/spin-http-js.wasm dist/spin.js
asset spin.js 4.57 KiB [emitted] (name: main)
runtime modules 670 bytes 3 modules
./src/index.ts 2.85 KiB [built] [code generated]
webpack 5.75.0 compiled successfully in 1026 ms
Starting to build Spin compatible module
Preinitiating using Wizer
Optimizing wasm binary using wasm-opt
Spin compatible module built successfully
Finished building all Spin components
If the build fails, check:
Are you in the hello_typescript directory?
Did you run npm install before building`?
Did you install the js2wasm plugin?
If you would like to know what build command Spin runs for a component, you can find it in the manifest, in the component.(id).build section:
[component.hello-typescript.build]command = "npm run build"
You can always run this command manually; spin build is a shortcut.
As a standard practice for Python, create and activate a virtual env:
If you are on a Mac/linux based operating system use the following commands:
$ python3 -m venv venv
$ source venv/bin/activate
If you are using Windows, use the following commands:
You can always run this command manually; spin build is a shortcut to save you having to remember it.
spin build can be used to build all components defined in the Spin manifest
file at the same time, and also has a flag that starts the application after
finishing the compilation, spin build --up.
Now that you have created the application and built the component, you can spin up
the application (pun intended):
$ spin up
Serving http://127.0.0.1:3000
Available Routes:
hello-typescript: http://127.0.0.1:3000 (wildcard)
If you would like to see what Spin is doing under the hood, set the RUST_LOG environment variable for detailed logs, before running spin up:
$ export RUST_LOG=spin=trace
The variable is RUST_LOG no matter what language your application is written in, because this is setting the log level for Spin itself.
Spin instantiates all components from the application manifest, and
creates the router configuration for the HTTP trigger according to the routes in the manifest. The
component can now be invoked by making requests to http://localhost:3000/
(or any path under that, since it’s a wildcard):
$ curl -i localhost:3000
HTTP/1.1 200 OK
content-type: text/plain
content-length: 14
date = "2023-11-04T00:00:01Z"
Hello, Fermyon
The curl output may vary based on which language SDK you use.
Congratulations! You just created, built and ran your first Spin application!
Deploy Your Application to Fermyon Cloud
spin up runs your application locally. Now it’s time to put it on the Web via Fermyon Cloud.
Fermyon Cloud’s Starter tier is free, and doesn’t require you to enter any kind of payment instrument. You only need a GitHub account.
Log in to Fermyon Cloud
Before deploying your application to Fermyon Cloud, you have to log in, using the spin login command. This generates a code to authorize your current device against the Fermyon Cloud, and prints a link that will take you to where you enter the code. (You will need to be logged into your GitHub account; if you’re not, it will prompt you to log in.) Follow the instructions in the prompt to complete the authorization process.
spin login prints a confirmation message when authorization completes:
$ spin login
Copy your one-time code:
XXXXXXXX
...and open the authorization page in your browser:
https://cloud.fermyon.com/device-authorization
Waiting for device authorization...
Device authorized!
Deploy the Application
Now let’s deploy the application:
$ spin deploy
The deployment process prints progress information as your application uploads and is rolled out to the cloud:
Uploading hello_typescript version 0.1.0+XXXXXXXX to Fermyon Cloud...
Deploying...
Waiting for application to become ready... ready
Available Routes:
hello-typescript: https://hello-typescript-XXXXXXXX.fermyon.app (wildcard)
You can Ctrl+Click on the link in the terminal to visit the web application you just deployed.
In the example output above, hello_typescript is a placeholder for your application name - you’ll see whatever you entered as the application name when you ran spin new earlier. The XXXXXXXX fragment is randomly generated to make a unique URL.
Congratulations again - you’ve now deployed your first Spin application to Fermyon Cloud!