From bed4d59e36cd1949ccb50b194058fda1330c3f4b Mon Sep 17 00:00:00 2001 From: Hayden Baker Date: Thu, 29 Aug 2024 12:24:18 -0700 Subject: [PATCH] Add tutorial for building a full-stack application with Smithy (#2362) --- docs/Makefile | 3 +- docs/source-2.0/index.rst | 1 + .../tutorials/full-stack-tutorial.rst | 908 ++++++++++++++++++ docs/source-2.0/tutorials/index.rst | 10 + 4 files changed, 921 insertions(+), 1 deletion(-) create mode 100644 docs/source-2.0/tutorials/full-stack-tutorial.rst create mode 100644 docs/source-2.0/tutorials/index.rst diff --git a/docs/Makefile b/docs/Makefile index 9e7bd25f6d9..48d1df58859 100644 --- a/docs/Makefile +++ b/docs/Makefile @@ -7,9 +7,10 @@ SHELL := /bin/bash install: requirements.txt python3 -m venv build/venv source build/venv/bin/activate && pip3 install -r requirements.txt . && pip3 install -e . + touch install clean: - -rm -rf build/* + -rm -rf build/* install html1: @source build/venv/bin/activate && $(SPHINXBUILD) -M html "source-1.0" "build/1.0" $(SPHINXOPTS) -W --keep-going -n $(O) diff --git a/docs/source-2.0/index.rst b/docs/source-2.0/index.rst index 7b5aafff260..6820afeb0d7 100644 --- a/docs/source-2.0/index.rst +++ b/docs/source-2.0/index.rst @@ -147,6 +147,7 @@ Read more spec/index trait-index guides/index + tutorials/index Additional specs aws/index ts-ssdk/index diff --git a/docs/source-2.0/tutorials/full-stack-tutorial.rst b/docs/source-2.0/tutorials/full-stack-tutorial.rst new file mode 100644 index 00000000000..e204651438d --- /dev/null +++ b/docs/source-2.0/tutorials/full-stack-tutorial.rst @@ -0,0 +1,908 @@ +====================== +Full Stack Application +====================== + +Overview +======== +For this tutorial, imagine we own a coffee shop. We would like to create a website for our customers to place an +order online, and be able to grab their coffee on the go. This application should show the available coffees, and +allow a customer to order a coffee. + +To build this application, we will walk you through using Smithy to define a model for the coffee service, generate +code for the client and server, and implement the front-end and back-end for the service. + +.. tip:: + This tutorial does not assume you are an expert in Smithy, but you may find it helpful to work through the + :doc:`../quickstart` before beginning this tutorial. + +Setting up the project +====================== +This application will consist of 4 major components: + +1. A model, which defines the service. +2. A server, which handles requests for coffee. +3. A client, which is generated from the model. +4. A web application, which uses the client to make requests to the server. + +-------------- +Pre-requisites +-------------- +To follow this tutorial, you will need to install a few tools: + +* :doc:`Smithy CLI <../guides/smithy-cli/cli_installation>` +* `Node.js (>= 16) `_ and `yarn `_ + +.. warning:: This project was made for Mac/Linux, it may not build correctly on Windows. + +------ +Set up +------ +Once you have the prerequisites installed, the Smithy CLI has a useful command to easily bootstrap projects from +`the repository of example smithy projects `_. Execute the following +command to set up the initial project: + +.. code-block:: sh + :caption: ``/bin/sh`` + + smithy init -t full-stack-application + +After running this command, you should have the project in the ``full-stack-application`` directory. +The directory tree should look like: + +.. code-block:: sh + :caption: ``directory structure`` + + full-stack-application + ├── Makefile + ├── README.md + ├── app + │ ├── ... + ├── client + │ ├── ... + ├── server + │ ├── ... + └── smithy + ├── ... + ... + +The ``README.md`` file contains important information about the project, like its directory structure and how to +build or run it. However, for this tutorial, we will show you which commands to run, when to run them, and +the expected outputs. + +Please run the following command to set the project up, so you can follow this tutorial: + +.. code-block:: sh + :caption: ``/bin/sh`` + + make init + +.. hint:: If you get stuck and want to completely start over, run ``make reset && make init``. + + Also, the complete project code for this tutorial is available + `here `_. + +Modeling the service +==================== +With the basic framework for the project established, let's walk through how to model our coffee service. +The service should provide a few capabilities: + +* Provide a menu of coffees. +* Provide the ability to order a coffee. +* Provide the ability to check the status of an order. + +------------------ +Adding the service +------------------ +The service shape is the entry-point of our API, and is where we define the operations our service exposes to a +consumer. First and foremost, let's define the initial service shape without any operations. Open the ``main.smithy`` +file and add the following: + +.. important:: For code blocks, the name of the current file is given in the top-left corner. + +.. code-block:: smithy + :caption: ``smithy/model/main.smithy`` + + $version: "2.0" + + namespace com.example + + use aws.protocols#restJson1 + + /// Allows users to retrieve a menu, create a coffee order, and + /// and to view the status of their orders + @title("Coffee Shop Service") + @restJson1 + service CoffeeShop { + version: "2024-08-23" + } + +We apply the ``@restJson1`` protocol trait to the service to indicate the service supports the +:doc:`../aws/protocols/aws-restjson1-protocol`. Protocols define the rules and conventions for serializing and +de-serializing data when communicating between client and server. + +------------- +Modeling data +------------- +Let's create basic representations of our data in Smithy. We will further refine our data model using +:ref:`traits `. Open the file titled ``coffee.smithy``. We will use it to write our definitions +of coffee-related structures: + +.. _full-stack-tutorial-operations: + +.. code-block:: smithy + :caption: ``smithy/model/coffee.smithy`` + + $version: "2.0" + + namespace com.example + + /// A enum describing the types of coffees available + enum CoffeeType { + DRIP + POUR_OVER + LATTE + ESPRESSO + } + + /// A structure which defines a coffee item which can be ordered + structure CoffeeItem { + @required + type: CoffeeType + + @required + description: String + } + + /// A list of coffee items + list CoffeeItems { + member: CoffeeItem + } + +------------------- +Modeling operations +------------------- +With the shapes defined above, let's create an operation for returning the menu to the consumer, and add it +to the service: + +.. code-block:: smithy + :caption: ``smithy/model/main.smithy`` + + ... + service CoffeeShop { + version: "2024-08-23" + operations: [ + GetMenu + ] + } + + /// Retrieve the menu + @http(method: "GET", uri: "/menu") + @readonly + operation GetMenu { + output := { + items: CoffeeItems + } + } + +We have named the operation ``GetMenu``. It does not define an input, and models its output as a structure with a single +member, ``items``, which contains ``CoffeeItems``, a shape we defined :ref:`above `. +With the ``restJson1`` protocol, the serialized response might look like the below: + +.. code-block:: json + :caption: ``GetMenuResponse (json)`` + + { + "items": [ + { + "type": "LATTE", + "description": "A creamier, milk-based drink made with espresso" + } + ] + } + +------------------- +Representing orders +------------------- +At this point, we still need to model the ordering functionality of our service. Let's modify the +``order.smithy`` file to hold definitions related to ordering. First, let's consider the following when +modeling an order: + +1. An order needs a unique identifier. +2. An order needs to have a status, such as "in-progress" or "completed". +3. An order needs to hold the coffee information (``CoffeeType``). + +With these requirements in mind, let's create the underlying data model: + +.. code-block:: smithy + :caption: ``smithy/model/order.smithy`` + + $version: "2.0" + + namespace com.example + + /// A unique identifier to identify an order + @length(min: 1, max: 128) + @pattern("^[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}$") + string Uuid + + /// An enum describing the status of an order + enum OrderStatus { + IN_PROGRESS + COMPLETED + } + +A universally unique identifier (or `"UUID" `_) should be +sufficient for our service. The order status is ``IN_PROGRESS`` (after submitting the order) or +``COMPLETED`` (when the order is ready). We will represent the coffee order information with the ``CoffeeType`` shape +we defined earlier. + +Let's compose these shapes together to create our representation of an order: + +.. code-block:: smithy + :caption: ``smithy/model/order.smithy`` + + /// An Order, which has an id, a status, and the type of coffee ordered + structure Order { + id: Uuid + coffeeType: CoffeeType + status: OrderStatus + } + +We're making great progress. However, if we think about an order and its `potential` set of operations +(`creating, reading, updating, deleting `_ an order), +there is tight relationship between the "state" of an order and its operations. Creating an order "begins" its +lifecycle, while deleting an order would "end" it. In Smithy, we encapsulate the relationship between an entity +and its operations with :ref:`resources `. Instead of the above structure, let's define an order "resource": + +.. code-block:: smithy + :caption: ``smithy/model/order.smithy`` + + /// An Order resource, which has a unique id and describes an order by the type of coffee + /// and the order's status + resource Order { + identifiers: { id: Uuid } + properties: { coffeeType: CoffeeType, status: OrderStatus } + read: GetOrder // <--- we will create this next! + create: CreateOrder // <--- we will create this next! + } + +With a resource, we attach an identifier, which uniquely identifies an instance of the resource. We use properties to +represent the state of an instance. In this case, we will only define a subset of the +:ref:`lifecycle operations ` to keep it simple (``create`` and ``read``). Let's define those now: + +.. code-block:: smithy + :caption: ``smithy/model/order.smithy`` + + /// Create an order + @idempotent + @http(method: "PUT", uri: "/order") + operation CreateOrder { + input := for Order { + @required + $coffeeType + } + + output := for Order { + @required + $id + + @required + $coffeeType + + @required + $status + } + } + + /// Retrieve an order + @readonly + @http(method: "GET", uri: "/order/{id}") + operation GetOrder { + input := for Order { + @httpLabel + @required + $id + } + + output := for Order { + @required + $id + + @required + $coffeeType + + @required + $status + } + + errors: [ + OrderNotFound // <--- we will create this next! + ] + } + +Since we are defining operations for a resource, we use :ref:`target elision ` by prefixing +members corresponding to the resource with ``$``. This reduces the amount of repetition when defining the input and +output shapes of an operation for a resource. + +When we define an operation which may return an explicit error, we should model it using the +:ref:`error trait `. Additionally, to refine our error, we will add the +:ref:`httpError trait ` to set a specific HTTP response status code when the service returns the error: + +.. code-block:: smithy + :caption: ``smithy/model/order.smithy`` + + /// An error indicating an order could not be found + @httpError(404) + @error("client") + structure OrderNotFound { + message: String + orderId: Uuid + } + +Now that we have defined an order resource and its operations, we need to attach the resource to the service: + +.. code-block:: smithy + :caption: ``smithy/model/main.smithy`` + + ... + service CoffeeShop { + ... + resources: [ + Order + ] + } + +Finally, you might be wondering why we did not model our coffee or menu as a resource. For our service, we are not +exposing any functionality related to the *lifecycle* of these entities. However, let's describe a hypothetical +example. We decide a coffee has properties like origin, roast, and tasting notes. Also, we choose to expose operations +for adding, updating, and removing coffees. In this case, coffee would be a prime candidate for modeling as a resource. + +Building the model +================== +The model for our coffee service is complete. Before we build the model, let's take a moment and learn how to configure +the build. The :ref:`smithy-build.json configuration file ` is how we instruct Smithy to build the +model. A :ref:`projection ` is a version of a model based on a set of :ref:`transformations `. +Plugins can be applied to a projection to produce artifacts based on its "version" of the model. +For our model, we will not configure any explicit projections, since Smithy always builds the ``source`` projection. +The ``source`` projection does not have any transformations applied, and its output includes the artifacts of +plugins applied at the root. To build the model, run: + +.. code-block:: sh + :caption: ``/bin/sh - smithy/`` + + smithy build + +.. hint:: For ``smithy`` commands, you should be under the ``full-stack-application/smithy`` directory. + + For ``make`` commands, you should be under the top-level directory (``full-stack-application``) + +Building the model will render artifacts under the ``build/smithy`` directory. Under it, The ``source`` directory +corresponds to the output (or "build artifacts") of the ``source`` projection. With the current configuration, Smithy +will produce the model in its :ref:`JSON AST representation `, and a ``sources`` directory which contains the +model files used in the build. Additional artifacts are produced by configuring plugins, and +:doc:`code-generators <../guides/using-code-generation/index>` are prime examples of this. + +Generating the server SDK +========================= +The server SDK is a code-generated component which provides built-in serialization, request-handling, and +scaffolding (or "stubs") for our service. It facilitates the implementation of the service by +providing these things, and allowing the implementer to focus on the business logic. Let's generate the server SDK +for our service by using the following build configuration: + +.. code-block:: json + :caption: ``smithy/smithy-build.json`` + + { + "version": "1.0", + "sources": ["model/"], + "maven": { + "dependencies": [ + "software.amazon.smithy:smithy-aws-traits:1.50.0", + "software.amazon.smithy:smithy-validation-model:1.50.0", + "software.amazon.smithy.typescript:smithy-aws-typescript-codegen:0.22.0" + ] + }, + "plugins": { + "typescript-ssdk-codegen": { + "package" : "@com.example/coffee-shop-server", + "packageVersion": "0.0.1" + } + } + } + +Run the build: + +.. code-block:: sh + :caption: ``/bin/sh - smithy/`` + + smithy build + +The build will should fail for the following reason: + +.. code-block:: text + :caption: ``failure message`` + + Projection source failed: software.amazon.smithy.codegen.core.CodegenException: + Every operation must have the smithy.framework#ValidationException error attached + unless disableDefaultValidation is set to 'true' in the plugin settings. + Operations without smithy.framework#ValidationException errors attached: + [com.example#CreateOrder, com.example#GetMenu, com.example#GetOrder] + + +The server SDK validates inputs by default, and enforces each operation has +the ``smithy.framework#ValidationException`` attached to it. We will fix this issue by attaching the error +to our service, meaning all operations in the service may return it. Let's do this now: + +.. code-block:: smithy + :caption: ``smithy/model/main.smithy`` + + use aws.protocols#restJson1 + use smithy.framework#ValidationException + + ... + service CoffeeShop { + ... + errors: [ + ValidationException + ] + } + + +After fixing this, run the build command again. The build should now succeed. The TypeScript code-generator +plugin will create a new artifact under ``build/smithy/source/typescript-ssdk-codegen``. This artifact contains +the generated server SDK (SSDK), which we will use in our back-end. + +Implementing the server +======================= +For this tutorial, we have included a ``Makefile``, which simplifies the process of building and running the +application. To use it, make sure to run ``make`` from the root of the application directory (where the ``Makefile`` +lives). Let's try it now: + +.. code-block:: sh + :caption: ``/bin/sh`` + + make build-server + +This command will run the code-generation for the server SDK, and then build the server implementation (which uses +the server SDK). The server package is located under the ``server`` directory, and contains +only two files under ``src``: + +* ``index.ts``: entry-point of the backend application, and where we initialize our service. +* ``CoffeeShop.ts``: implementation of a `CoffeeShopService` from the generated server SDK. + +The ``ssdk`` directory is a link to our generated server SDK, which is an output of the smithy build. This is where +the server imports the generated code from. Let's take a look at the core of the coffee shop implementation: + +.. code-block:: TypeScript + :caption: ``server/src/CoffeeShop.ts`` + + // An implementation of the service from the SSDK + export class CoffeeShop implements CoffeeShopService { + ... + + async CreateOrder(input: CreateOrderServerInput, context: CoffeeShopContext): Promise { + console.log("received an order request...") + // TODO: Implement me! + return; + } + + async GetMenu(input: GetMenuServerInput, context: CoffeeShopContext): Promise { + console.log("getting menu...") + // TODO: Implement me! + return; + } + + async GetOrder(input: GetOrderServerInput, context: CoffeeShopContext): Promise { + console.log(`getting an order (${input.id})...`) + // TODO: Implement me! + return; + } + + ... + } + +These three methods are how we implement the core business logic of the service. They are exposed by the +``CoffeeShopService`` interface exported by the server SDK. This file already contains some of the underlying logic +for how our implementation will run: there is an orders queue, an orders map, and an order-handling procedure +(``handleOrders``). We will use these to implement the operations for our service. Let's start with the simplest +operation, ``GetMenu``. We will modify the operation to return a menu containing one coffee item for +each type of coffee: + +.. code-block:: TypeScript + :caption: ``server/src/CoffeeShop.ts`` + + async GetMenu(input: GetMenuServerInput, context: CoffeeShopContext): Promise { + console.log("getting menu...") + return { + items: [ + { + type: CoffeeType.DRIP, + description: "A clean-bodied, rounder, and more simplistic flavour profile.\n" + + "Often praised for mellow and less intense notes.\n" + + "Far less concentrated than espresso." + }, + { + type: CoffeeType.POUR_OVER, + description: "Similar to drip coffee, but with a process that brings out more subtle nuances in flavor.\n" + + "More concentrated than drip, but less than espresso." + }, + { + type: CoffeeType.LATTE, + description: "A creamier, milk-based drink made with espresso.\n" + + "A subtle coffee taste, with smooth texture.\n" + + "High milk-to-coffee ratio." + }, + { + type: CoffeeType.ESPRESSO, + description: "A highly concentrated form of coffee, brewed under high pressure.\n" + + "Syrupy, thick liquid in a small serving size.\n" + + "Full bodied and intensely aromatic." + } + ] + } + } + +For our menu, we have added a distinct item and description for each of our coffee enumerations (``CoffeeType``). +For ordering, we will maintain an order map to simulate a database that stores historical order information, +and an order queue to keep track of in-flight orders. The ``handleOrders`` method processes in-flight orders +and updates this queue. Let's implement order submission, or ``CreateOrder``: + +.. code-block:: TypeScript + :caption: ``server/src/CoffeeShop.ts`` + + async CreateOrder(input: CreateOrderServerInput, context: CoffeeShopContext): Promise { + console.log("received an order request...") + const order = { + orderId: randomUUID(), + coffeeType: input.coffeeType, + status: OrderStatus.IN_PROGRESS + } + + context.orders.set(order.orderId, order) + context.queue.push(order) + + console.log(`created order: ${JSON.stringify(order)}`) + return { + id: order.orderId, + coffeeType: order.coffeeType, + status: order.status + } + } + +After submitting an order, we can retrieve its information from the order map. This information should be retrievable +through the ``GetOrder`` operation. Let's implement it now: + +.. code-block:: TypeScript + :caption: ``server/src/CoffeeShop.ts`` + + async GetOrder(input: GetOrderServerInput, context: CoffeeShopContext): Promise { + console.log(`getting an order (${input.id})...`) + if (context.orders.has(input.id)) { + const order = context.orders.get(input.id) + return { + id: order.orderId, + coffeeType: order.coffeeType, + status: order.status + } + } else { + console.log(`order (${input.id}) does not exist.`) + throw new OrderNotFound({ + message: `order ${input.id} not found.`, + orderId: input.id + }) + } + } + +With these operations implemented, our server is fully implemented. Let's build and run it: + +.. code-block:: sh + :caption: ``/bin/sh`` + + make run-server + +This command will build and run the server. You should see the following output: + +.. code-block:: text + :caption: output + + Started server on port 3001... + handling orders... + +With the server running, let's test it by sending it a request. Open a new terminal and send a request to the ``/menu`` +route using ``cURL``. This will send a request to the server, and the server should handle it with +the ``GetMenu`` operation: + +.. code-block:: sh + :caption: ``/bin/sh`` + + curl localhost:3001/menu + +You should see the output of the ``GetMenu`` operation that we implemented above. You may stop the server with +``CTRL + C`` in the terminal where it is running. With the server implemented, we will move on to the client. + +Generating the client +===================== + +To run the code-generation for the client, we will add another plugin to the ``smithy-build.json`` configuration file: + +.. code-block:: json + :caption: ``smithy/smithy-build.json`` + + { + // ... + "plugins": { + "typescript-ssdk-codegen": { + "package" : "@com.example/coffee-shop-server", + "packageVersion": "0.0.1" + }, + // add the client codegen plugin + "typescript-client-codegen": { + "package": "@com.example/coffee-shop-client", + "packageVersion": "0.0.1" + } + } + } + +Run the build: + +.. code-block:: sh + :caption: ``/bin/sh - smithy/`` + + smithy build + +Similar to the server SDK, Smithy will generate the TypeScript client artifacts under the +``build/smithy/source/typescript-client-codegen`` directory. We will use this client to make calls to our backend +service. + +Using the client +================ +Like with the server, there is a make target for generating and building the TypeScript client. Let's try it now: + +.. code-block:: sh + :caption: ``/bin/sh`` + + make build-client + +This command will code-generate the client with Smithy, and then build the generated TypeScript package. The command +will link the client in the project root under ``client/sdk``. To use the client ad-hoc, run the following command: + +.. code-block:: sh + :caption: ``/bin/sh`` + + make repl-client + +This command launches a TypeScript `REPL `_ with +the generated client installed. Before we use the generated client, we must run the server. Without the server running, +the client will not be able to connect. In another terminal, launch the server with the following command: + +.. code-block:: sh + :caption: ``/bin/sh`` + + make run-server + +With the server running, we will instantiate and use the client. In the terminal running the REPL, run the +following: + +.. code-block:: TypeScript + :caption: ``repl`` + + import { CoffeeShop } from '@com.example/coffee-shop-client' + + const client = new CoffeeShop({ endpoint: { protocol: 'http', hostname: 'localhost', port: 3001, path: '/' } }) + + await client.getMenu() + +Like when we tested the server with ``cURL``, you should see the output of the ``GetMenu`` operation we implemented. +Let's try submitting an order: + +.. code-block:: TypeScript + :caption: ``repl`` + + import { CoffeeType } from '@com.example/coffee-shop-client' + + await client.createOrder({ coffeeType: "DRIP" }) + +After creating the order, you should get response like: + +.. code-block:: typescript + :caption: response + + { + '$metadata': { + // metadata, such as response code, added by the client + }, + coffeeType: 'DRIP', // <--- the type of coffee we ordered + id: 'ee97e900-d8dd-4770-904c-3d175cda90c3', // <--- the order id + status: 'IN_PROGRESS' // <--- the order status + } + +The order should be ready by the time you submit this next command. Let's retrieve the order: + +.. code-block:: TypeScript + :caption: ``repl`` + + await client.getOrder({ id: '' }) // <--- make sure to replace with your id + +Once you execute the command, you should see your order information: + +.. code-block:: typescript + :caption: response + + { + '$metadata': { + // ... + }, + coffeeType: 'DRIP', // <--- the type of coffee we ordered + id: 'ee97e900-d8dd-4770-904c-3d175cda90c3', // <--- the order id + status: 'COMPLETED' // <--- the order status, which should be 'COMPLETED' + } + +You may stop the REPL and server with ``CTRL + C`` in the respective terminals. We have +tested each operation we implemented in the server using the generated client, and verified both the client +and server communicate with each other. + +------------------ +In the application +------------------ +Using the client in the application is not much different from what we just did. + +In the ``app`` directory, there is a file, ``app/index.ts``, which contains code that instantiates and uses the +client. First, we create the client, and then we create helper methods to use the client: + +.. code-block:: TypeScript + :caption: ``app/app/index.ts`` + + import { CoffeeItem, CoffeeShop, CoffeeType, OrderStatus } from "@com.example/coffee-shop-client"; + + ... + // create a coffee service client singleton and getter + let client: CoffeeShop + export function getClient(): CoffeeShop { + return client || (client = new CoffeeShop({ + endpoint: { + protocol: "http", + hostname: "localhost", + port: 3001, + path: "/" + } + })); + } + + // coffee service client helpers ------ + export async function getMenuItems(): Promise { + let items: CoffeeItem[] = [] + try { + const res = await getClient().getMenu(); + items = res.items || [] + } catch (err) { + console.log(err) + } + return items + } + ... + +We use these helper methods in our application to make requests to the server: + +.. code-block:: TypeScript + :caption: ``app/components/Menu.tsx`` + + ... + import MenuItem from "@/components/MenuItem"; + import { CoffeeItem } from "@com.example/coffee-shop-client"; + + const Menu = async () => { + let menuItems: CoffeeItem[] = await getMenuItems(); + ... + +Running the application +======================= +Since we know how to generate and use the client and server, let's put it all together to use with the web application. +The application exists under the ``app`` directory. To build the application, use the ``build-app`` make target. +The application will run when using the ``run-app`` target. Since this application uses the generated client to make +requests, the server must be run alongside the app. For convenience, you may run both the web application and +the server in the same terminal: + +.. important:: If you are already running the server, stop it before continuing past this point. + +.. code-block:: sh + :caption: ``/bin/sh`` + + make run + +While running the application in this way is convenient, it will intertwine the output of the application and server. +If you would like to keep them separate, you should run the other targets (``run-server`` and ``run-app``). +Using the method of your choice, launch the server and the application. + +Launch your browser and open http://localhost:3000. You should see the coffee shop web application. +Try ordering a coffee. When interacting with this application, you should see logs for both the client and server. +While this application is simple, it shows how to integrate a smithy-generated client with an +application running in the browser. You may stop the application in the terminal and close the browser. + +Making a change (optional) +========================== +We would like to add a new coffee to our menu. The new menu item should have the following details: + +* type: COLD_BREW +* description: A high-extraction and chilled form of coffee that has been cold-pressed. + Different flavor profile than other hot methods of brewing. + Smooth and slightly more caffeinated as a result of its concentration. + +.. note:: Before you proceed to the solution, try making the changes needed by yourself. + +.. raw:: html + +
+ Solution + +To add a new coffee, we will first make a change to our model. We need to add a new value for the ``CoffeeType`` +enumeration: + +.. code-block:: smithy + :caption: ``smithy/model/coffee.smithy`` + + /// An enum describing the types of coffees available + enum CoffeeType { + DRIP + POUR_OVER + LATTE + ESPRESSO + COLD_BREW + } + +Next, we need to update the server code to add a new item to the menu. First, we should build the model and run the +code-generation for the server SDK to generate the new value. Run ``make build-ssdk``. After re-generating the +server SDK, we will make the change to the implementation of ``GetMenu``. We will use the new value and the +description above to add a new item to the menu: + +.. code-block:: TypeScript + :caption: ``server/src/CoffeeShop.ts`` + + async GetMenu(input: GetMenuServerInput, context: CoffeeShopContext): Promise { + console.log("getting menu...") + return { + items: [ + ... + { + type: CoffeeType.COLD_BREW, + description: "A high-extraction and chilled form of coffee that has been cold-pressed..\n" + + "Different flavor profile than other hot methods of brewing.\n" + + "Smooth and slightly more caffeinated as a result of its concentration." + } + ] + } + } + +Now, make a similar change in the web application code to render a new image for the new type of coffee: + +.. code-block:: TypeScript + :caption: ``app/app/index.ts`` + + ... + case CoffeeType.COLD_BREW: + return "/cold-brew.png" + default: + ... + +Finally, we will run the whole application to see the changes (``make run``). After you run it and open +http://localhost:3000 in your browser, you should see the new menu item in the web application. + +.. raw:: html + +
+ +Wrapping up +=========== +In this tutorial, you used Smithy to build a full-stack application for a simple coffee shop. You wrote a Smithy model +for a service based on a list of requirements. Afterward, you configured Smithy using the ``smithy-build.json`` +configuration. You added plugins to code-generate a TypeScript server SDK and client. You implemented the +service using the server SDK, and made requests to it using the client. Finally, you used the client in the web +application to make requests from within the browser to our service. + +--------- +What now? +--------- +We covered several topics in this tutorial, but there is still so much to learn. For other examples of smithy projects, +please see the following repositories: + +* `awesome-smithy `_: A list of projects based in the smithy ecosystem +* `smithy-examples `_: A repository of example smithy projects diff --git a/docs/source-2.0/tutorials/index.rst b/docs/source-2.0/tutorials/index.rst new file mode 100644 index 00000000000..8323d6067a0 --- /dev/null +++ b/docs/source-2.0/tutorials/index.rst @@ -0,0 +1,10 @@ +========= +Tutorials +========= + +The goal of these pages is to help you learn by doing. While tutorials will have explicit objectives (e.g. "build a full-stack application"), the true intention is that you learn how to use Smithy holistically. You will gain an understanding of various tools, concepts, interactions, and how they fit together in this ecosystem. + +.. toctree:: + :maxdepth: 1 + + full-stack-tutorial