Central Coast Cauldrons is a stubbed out API meant to serve as a starting point for learning how to build backend servies that integrate with a persistance layer. You will progressively build out your own forked version of the API and integrate with a progressively more sophisticated database backend. When you register your backend at the Consortium of Concotions and Charms simulated customers will shop at your store using your API.
The application's setting is a simulated fantasy RPG world with adventurers seeking to buy potions. You are one of many shops in this world that offer a variety (over 100k possibilities) of potions.
With an initial capital of 100 gold, no potions in your inventory, and devoid of barrels, your backend API is scheduled to be invoked at regular intervals, known as 'ticks' (set at every 5 minutes while testing, but I will later make this happen every 2 hours and align with calendar-time). There are 12 ticks in a day, and 7 days in a week.
There are three primary actions that may unfold during these ticks:
-
Customer Interactions: On each tick, one or more simulated customers access your catalog endpoint intending to buy potions. The frequency and timing of customer visits vary based on the time of day, and each customer exhibits specific potion preferences. Your shop's performance is evaluated and scored based on multiple criteria (more details on Consortium of Concotions and Charms), which in turn influences the frequency of customer visits.
-
Potion Creation: Every alternate tick presents an opportunity to brew new potions. Each potion requires 100 ml of either red, green, blue, or dark liquid. You must have sufficient volume of the chosen color in your barrelled inventory to brew a potion.
-
Barrel Purchasing: On every 12th tick, you have an opportunity to purchase additional barrels of various colors. Your API receives a catalog of barrels available for sale and should respond with your purchase decisions. The gold cost of each barrel is deducted from your balance upon purchase.
Part of the challenge in these interactions is you are responsible for keeping track of your gold and your various inventory levels. The Consortium of Concotions and Charms separately keeps an authoritiative record (which can be viewed under Shop stats).
Customers of various types have different seasonality on when they show up. For example, some customers are more likely to shop on certain days of the week and at certain times of day. Customers each have their own class which has a huge impact on what types of potions that customer is looking for. The amount a customer is willing to spend on a given potion depends on both the customers own level of wealth and how precisely the potions available in a store match their own preference.
Lastly, customers are more likely to shop in a store in the first place that has a good reputation. You can see your shop's reputation with a particular class at Consortium of Concotions and Charms. Reputation is based on three different factors:
- Value: Value is calculated as the delta between the utility the customer gets from a potion (which is increased by getting as close to possible to that customer's preference) and how much the store charges. The highest value is achieved by offering cheap potions that exactly match a customer's preference.
- Reliability: Reliability is based upon not having errors in the checkout process. Your site being down or offering up potions for sale you don't have in inventory are examples of errors that will hurt reliability.
- Recognition: Recognition is based upon the number of total successful purchasing trips that customers of that class have had. The more you serve a particular class, the more others of that class will trust to come to you.
For more information please reference the API Spec
Follow these steps to get your potion shop up and running:
- Create your own Github repository based on Central Coast Cauldrons GitHub repository. When creating your repository, name it something unique from centralcoastcauldrons.
- Register on Render. Click New + and select Web Service. We will do a basic Build and deploy from a Git repository. Connect it to your Github repository and point it the new repository you created on step 1.
- Name your service something cute and unique. Oregon is fine as a region. Leave branch and Root Directory as default. The Runtime should be Python 3. Build command should be pip install -r requirements.txt. Start command should be changed to uvicorn src.api.server:app --host 0.0.0.0 --port $PORT. Select the Free Instance Type. Seelct Advanced and add two environment variables to start. The first is called
API_KEY
. Assign a unique string value to this variable. This string acts as a unique identifier for your shop and helps secure your communications. You will use it later when testing your service and registering it. The second is calledPYTHON_VERSION
. We will set this to 3.11.4. - Visit your newly deployed project to ensure it's functioning as expected. Try navigating to https://
your-project
.onrender.com/docs. Once there, click the 'Authorize' button in the upper right corner and enter the sameAPI_KEY
you entered into environment variables earlier. After authorization, try out various endpoints to confirm their functionality. - Navigate to Consortium of Concotions and Charms, sign in using your GitHub account, and add your newly created shop to the platform. Be sure to provide the URL of your newly deployed webservice (don't include doc/, just the base url) and the
API_KEY
you set earlier. - Return to Consortium of Concotions and Charms to monitor the next tick. Check for changes in your gold balance, potion inventory, and other assets. You should see with even this purely static implementation of your API barrels being purchased, potions getting mixed, and selling some potions to customers.
The first version of your improved store will simply keep track of how many red potions are available. Follow these steps:
- Create an account on Supabase.
- We will start with the simplest possible schema: a single row in a single table. Create a new project in Supabase, select the 'Table Editor' on the left-hand navigation menu, and then click the "Create a new table" button. Create a table called
global_inventory
. This table should have columns namednum_red_potions
(int4),num_red_ml
(int4), andgold
(int4) to keep track of your current resources. - Insert an initial row in your database and set
num_red_potions
to 0,num_red_ml
to 0, andgold
to 100. - Add your database connection details to the environment variables in your Vercel project. Within your Supabase project settings, go to Database -> Connection String -> URI, copy the connection string, and replace
PASSWORD
with your database password. Back in Vercel, add this modified string as a new environment variable namedPOSTGRES_URI
.
In your backend repository, establish a connection to your Supabase database using SQLAlchemy, drawing the POSTGRES_URI
from environment variables. To do so, create a database.py file in your src folder with the following code:
import os
import dotenv
from sqlalchemy import create_engine
def database_connection_url():
dotenv.load_dotenv()
return os.environ.get("POSTGRES_URI")
engine = create_engine(database_connection_url(), pool_pre_ping=True)
Next, we will leverage the global_inventory
table to keep tabs on potion quantities and the total liquid volume in your inventory. Adapt your API to deliver updated JSON responses based on your inventory while maintaining the same API structure. For the sake of simplicity, assume you're only brewing and selling red potions at this stage.
In your endpoint definition files add these imports:
import sqlalchemy
from src import database as db
And execute SQL in each of your endpoints (barrels.py, bottler.py, carts.py, and catalog.py) as:
with db.engine.begin() as connection:
result = connection.execute(sql_to_execute)
You will need to use SELECT and UPDATE sql statements. In this version you won't need to do any INSERTs or DELETEs.
As a very basic initial logic, purchase a new small red potion barrel only if the number of potions in inventory is less than 10. Always mix all available red ml if any exists. Offer up for sale in the catalog only the amount of red potions that actually exist currently in inventory.
Once you've finished making your changes, go back to Consortium of Concotions and Charms and click "Burn Shop to Ground!" at the bottom of the page to reset your shop's state back to the beginning.
With the release of this version, you should no longer encounter job errors resulting from attempting to buy barrels without sufficient gold, mix potions without the necessary ml of ingredients, or sell potions not currently in your inventory.