Making REST APIs Typesafe With feTS
Aleksandra Sikora, @aleksandrasays
whoami
- open-source engineer @
- π§ββοΈ
- org of WrocΕaw
previously
- .js maintainer
- cc tech lead
π¦ @aleksandrasays
π @beerose
π https://aleksandra.codes
data:image/s3,"s3://crabby-images/2fb81/2fb81678c0a2ffd7d5f3767f21a9a04e4ebbc01c" alt=""
data:image/s3,"s3://crabby-images/3cd0f/3cd0f88bffedb5d4f59e3b80ce8261d255932f13" alt=""
data:image/s3,"s3://crabby-images/0ebbc/0ebbc3cb343dea831b0c6285a5015f74ffde56ee" alt=""
data:image/s3,"s3://crabby-images/78063/780638c26186ab80832cc9e0a7371a9b8f503053" alt=""
data:image/s3,"s3://crabby-images/1b344/1b344c82b206cc5019bd412b7eab8a360da96050" alt=""
data:image/s3,"s3://crabby-images/77ac5/77ac5760e6e27724b16312cf2585c03f482bcaf1" alt=""
What is an API?
data:image/s3,"s3://crabby-images/b76a3/b76a3bcea19fd05c62569f20116a22aee4aeef7e" alt=""
Why are we talking about APIs?
Server & Client
Server & Client
π₯΄
Server & Client
π
API Layer Problems
Boilerplate
Lost typesafety
Repetitive error handling
data:image/s3,"s3://crabby-images/a312b/a312bb6d478281b2d22175cdbc5b37a4214e7535" alt=""
TYPE-SAFE
TYPE-SAFE
data:image/s3,"s3://crabby-images/a03ab/a03ab8b6508db5f99f3caf4ba5140a768dba17ab" alt=""
RPC
1981
data:image/s3,"s3://crabby-images/a5cc9/a5cc9e370faada8d4bffe7590b8848849681a19a" alt=""
What is RPC?
// one-computer.js
function welcome(name) {
return `Hello, ${name}!`
}
const greeting = welcome("Helsinki!")
// ^ "Hello, Helsinki!"
// server.js
function welcome(name) {
return `Hello, ${name}!`
}
startImaginaryServer({ welcome })
// client.js
const greeting = await fetch(
`https://aleksandra.says/rpc/welcome`,
{ body: JSON.stringify("Helsinki") }
)
What is RPC?
calling remote procedures as if they were local
π
π₯΄
- Client & server tightly coupled
- Having to use the same language
- Need to learn all the procedure names
- Having to use multi-threaded servers
- Parameters marshalling
- Exception handling
Problems with RPC
RPC -> non-agnostic
RPC -> non-agnostic
βββββ
now we're looking for sth
CORBA
1991
NOT ΓORBA
data:image/s3,"s3://crabby-images/eef8a/eef8a25d97a9925c8907a2a06a999d93c0060d49" alt=""
data:image/s3,"s3://crabby-images/9082b/9082b74a0b655ab43bba393e133cf97eb82bb523" alt=""
AND NOT COBRA
module Finance {
typedef sequence<string> StringSeq;
struct AccountDetails {
string name;
StringSeq address;
long account_number;
double current_balance;
};
exception insufficientFunds { };
interface Account {
void deposit(in double amount);
void withdraw(in double amount)
raises(insufficientFunds);
readonly attribute AccountDetails details;
};
};
IDL
data:image/s3,"s3://crabby-images/04579/045795f8122a2e32ef45d1fdcb208a2c349517d4" alt=""
Developer trying to learn Corba
- Complexity
- Steep learning curve
- Mapping problems
- Name confused with a poisonous snake
Problems with CORBA
CORBA -> complex
simple
- -
CORBA -> complex
now we need
SOAP
1998
<?xml version="1.0"?>
<soap:Envelope
xmlns:soap="http://www.w3.org/2003/05/soap-envelope/"
soap:encodingStyle="http://www.w3.org/2003/05/soap-encoding">
<soap:Body>
<m:GetUserResponse>
<m:Username>Tony Stark</m:Username>
</m:GetUserResponse>
</soap:Body>
</soap:Envelope>
<?xml version="1.0"?>
<soap:Envelope xmlns:soap="http://www.w3.org/2003/05/soap-envelope">
<soap:Header>
</soap:Header>
<soap:Body>
<m:GetUser>
<m:UserId>123</m:UserId>
</m:GetUser>
</soap:Body>
</soap:Envelope>
π βοΈ
- Heavy, requires more bandwidth
- POST = no cache on HTTP layer
- Tightly coupled with server
- Inflexible
Problems with SOAP
SOAP -> heavy
light
-
SOAP -> heavy
now we're looking for sth
REST
2000
When the web started to change
Can request and update resources
Exposes resources
data:image/s3,"s3://crabby-images/197ff/197ff725bcc300f41c20baf11fbc22de386172cc" alt=""
data:image/s3,"s3://crabby-images/bdc20/bdc20d5cfb654b22e17e9004d8c04113a59c8f82" alt=""
Operation | RPC | REST |
---|---|---|
Login | POST /login | POST /sessions |
Logout | POST /logout | DELETE /sessions |
Get user by id | GET /getUser?id=123 | GET /users/123 |
Get user's todo items | GET /getTodos?userId=123 | GET /users/123/todos |
Add new todo item | POST /addTodo | POST users/123/todos |
Update todo item | POST /updateTodo | PUT /todos/1 |
Delete todo item | POST /deteteTodo | DELETE /todos/1 |
RPC vs. REST
JSON-RPC
RESTful
"REST"
- Over fetching
- Big payloads
- n+1 problem
- Limiting constraints
- No end-to-end typesafety
Problems with REST
REST -> inflexible
-
REST -> inflexible
time for something
GraphQL
2012
REST API
GraphQL API
API
App
GET users/
GET tasks/
GET tags/
API
App
POST graphql/
Body:
{ "query": "query { users {...} }" }
vs
Client controls the data it gets
User 1
Task 1
Task 2
Tag 1
Tag 2
query {
user(id: 1) {
name
tasks {
name
status
tags {
id
}
}
}
}
name
surname
age
status
name
priority
name
priority
status
description
id
Tag 3
id
description
id
description
id
data:image/s3,"s3://crabby-images/d39cf/d39cff4fed9e625fe51014a71506fba1a95d42a3" alt=""
- Same POST-caching problem as in SOAP
- You have to generate types
- If you use tools like Hasura,
you push a lot of domain logic to frontend - Otherwise β boilerplate!
Problems with GraphQL
at least until stuff like Max Stoiber's GraphQL CDN popped up
GraphQL -> extra work & type-safety
-
GraphQL -> extra work & type-safety
-
-
for free and out of the box
RPC
2020
Revisiting the original promise of RPC
1981
data:image/s3,"s3://crabby-images/27b73/27b73dccc4fc4f4f03370346cc81cc3ac23e7616" alt=""
data:image/s3,"s3://crabby-images/7e028/7e02815ea96d320f0e5cedf7ddb068ce417c576f" alt=""
Fullstack TypeScript app
Fullstack TypeScript app
Fullstack TypeScript app
tRPC query & mutation procedures
Remix loader pattern
React Server Components
Qwik City
Blitz RPC query & mutation resolvers
pRPC
Source: https://twitter.com/markdalgleish/status/1256800146118959109
data:image/s3,"s3://crabby-images/90729/907290eb512453465cb1c914f53eb76764de93b7" alt=""
?
2023
Cool, but...
again, non-agnostic
once again, we'd like something agnostic
type-safety is a must
data:image/s3,"s3://crabby-images/67e4b/67e4b08f42f1486b3259b8886c9ec1d1eab6dda9" alt=""
data:image/s3,"s3://crabby-images/80e19/80e19355177fcff44195d74f9bda71e9a3f744d6" alt=""
data:image/s3,"s3://crabby-images/d8ad5/d8ad568dd8b4526150fc5196546f9015e7e0b677" alt=""
data:image/s3,"s3://crabby-images/227a7/227a778e0062069bb32aac2eef0549561bca93ac" alt=""
feTS
DEMO
π Type-safety out of the box
π No runtime overhead
π IDE features
https://the-guild.dev/openapi/fets
data:image/s3,"s3://crabby-images/4274f/4274f991396776a7ceaa1ad6200f62f993e790ff" alt=""
Summary
data:image/s3,"s3://crabby-images/32373/323736fba365704aa315a3f9503ff4bcfcfe315e" alt=""
data:image/s3,"s3://crabby-images/4d27c/4d27c7b4167ba62f5fbc9a027d234dc266d2a621" alt=""
APIs
data:image/s3,"s3://crabby-images/6284f/6284f51c6cdaa126c6ec622f6a38efbd395e70ec" alt=""
data:image/s3,"s3://crabby-images/511c0/511c0faca0d24372a9ed5d19b836a92091cc5a9c" alt=""
Thank you!
@aleksandrasays
www.aleksandra.codes
data:image/s3,"s3://crabby-images/1c54f/1c54f8bbbc6c4124eaf1ead146e7354c54ce4a3b" alt=""
OLX Meetup: Typesafe REST with feTS
By Aleksandra Sikora
OLX Meetup: Typesafe REST with feTS
- 1,398