TeamSnap
&
GraphQL
Silver bullet or over-hyped technology?
Dustin McCraw
10/11/19
Agenda:
-
15 minutes on the challenges facing our platform
-
15 minutes explain what GraphQL is
and if it can address those problems
- 15 minutes for demos and discussion
I need your help!
I'm really looking to either disprove or validate these ideas.
I want to either be able to pursue these ideas or drop them and so I need your help to validate these solutions or
find issues that I haven't thought about.
This presentation will be a success if...
You come away with an understand of some of the challenges facing our platform and some ways to solve them.
Api's and Client's
In 5 years I've written code for:
Snapi, McFeely, Dozer,
org-api, org-service, soylent, accounts-api
classic, league-frontend, org-frontend,
payments-frontend, NextJenn
I have worked all over the stack at TeamSnap and I have seen some of challenges when it comes to developing and consuming our REST apis.
Challenge #1
Challenge #1
Challenge #1
Challenge #1
How do clients interact with our Microservices Architecture?
- Are they expected to know each of them individually?
- Mobile already has to connect to Snapi, Hoyle and Payments-Api.
- What about when they need to connect to accounts-api or org-api?
Challenge #2
Snapi 8 events
Snapi 0 events
2957 lines of json
40,192 bytes
412 lines of json
6,356 bytes
Challenge #2
How large can we expect our JSON payloads?
collection+json
With 80% of our DAU's coming from mobile,
payload size is incredibly import.
Our users can be on 4G, or even worse 3G.
The smaller payload we send the better.
Clients have to parse all of that code, the more they parse, the slower they perform.
Challenge #3
{
"collection": {
"version": "3.822.1",
"href": "https://api.teamsnap.com/v3/events",
"rel": "events",
"template": {
"data": [
{
"name": "type",
"value": "event"
},
{
"name": "additional_location_details",
"value": null
},
{
"name": "browser_time_zone",
"value": null
},
{
"name": "division_location_id",
"value": null
},
{
"name": "doesnt_count_towards_record",
"value": null
},
{
"name": "duration_in_minutes",
"value": null
},
{
"name": "game_type_code",
"value": null
},
{
"name": "icon_color",
"value": null
},
{
"name": "is_canceled",
"value": null
},
{
"name": "is_game",
"value": null
},
{
"name": "is_overtime",
"value": null
},
{
"name": "is_shootout",
"value": null
},
{
"name": "is_tbd",
"value": null
},
{
"name": "label",
"value": null
},
{
"name": "location_id",
"value": null
},
{
"name": "minutes_to_arrive_early",
"value": null
},
{
"name": "name",
"value": null
},
{
"name": "notes",
"value": null
},
{
"name": "notify_opponent",
"value": null
},
{
"name": "notify_opponent_contacts_email",
"value": null
},
{
"name": "notify_opponent_contacts_name",
"value": null
},
{
"name": "notify_opponent_notes",
"value": null
},
{
"name": "notify_team",
"value": null
},
{
"name": "notify_team_as_member_id",
"value": null
},
{
"name": "opponent_id",
"value": null
},
{
"name": "points_for_opponent",
"value": null
},
{
"name": "points_for_team",
"value": null
},
{
"name": "repeating_include",
"value": null,
"prompt": "When updating a repeating event, this is a required field. Values are: \"all\" - updates all events in this series, \"future\" - updates this event and all that occur after, \"none\" - only updates a single event."
},
{
"name": "repeating_type_code",
"value": null,
"prompt": "A code for the frequency of the repeated event, this is required with the \"repeating_include\" attribute when creating a repeating event. Valid values are: \"1\" - repeat an event daily, \"2\" - repeat an event weekly."
},
{
"name": "repeating_until",
"value": null,
"prompt": "A date when the repeating event should end, this is inclusive so an event will be created on this day if it falls before the next event specified by \"repeating_type_code\". This attribute is required with \"repeating_type_code\" when creating a repeating event."
},
{
"name": "results",
"value": null
},
{
"name": "results_url",
"value": null
},
{
"name": "shootout_points_for_opponent",
"value": null
},
{
"name": "shootout_points_for_team",
"value": null
},
{
"name": "start_date",
"value": null
},
{
"name": "team_id",
"value": null
},
{
"name": "time_zone",
"value": null
},
{
"name": "tracks_availability",
"value": null
},
{
"name": "uniform",
"value": null
}
]
},
"links": [
{
"rel": "assignments",
"href": "https://api.teamsnap.com/v3/assignments"
},
{
"rel": "availabilities",
"href": "https://api.teamsnap.com/v3/availabilities"
},
{
"rel": "division_location",
"href": "https://api.teamsnap.com/v3/division_locations"
},
{
"rel": "event_lineups",
"href": "https://api.teamsnap.com/v3/event_lineups"
},
{
"rel": "event_statistics",
"href": "https://api.teamsnap.com/v3/event_statistics"
},
{
"rel": "location",
"href": "https://api.teamsnap.com/v3/locations"
},
{
"rel": "opponent",
"href": "https://api.teamsnap.com/v3/opponents"
},
{
"rel": "statistic_data",
"href": "https://api.teamsnap.com/v3/statistic_data"
},
{
"rel": "team",
"href": "https://api.teamsnap.com/v3/teams"
},
{
"rel": "root",
"href": "https://api.teamsnap.com/v3/"
},
{
"rel": "self",
"href": "https://api.teamsnap.com/v3/events/search?is_game=true&page_number=1&page_size=1&sort_start_date=desc&started_before=2019-10-01T19:18:45-07:00&team_id=6618318"
},
{
"rel": "first",
"href": "https://api.teamsnap.com/v3/events/search?is_game=true&page_number=1&page_size=1&sort_start_date=desc&started_before=2019-10-01T19%3A18%3A45-07%3A00&team_id=6618318"
},
{
"rel": "last",
"href": "https://api.teamsnap.com/v3/events/search?is_game=true&page_number=0&page_size=1&sort_start_date=desc&started_before=2019-10-01T19%3A18%3A45-07%3A00&team_id=6618318"
}
],
"queries": [
{
"rel": "search",
"sort": [
"start_date"
],
"href": "https://api.teamsnap.com/v3/events/search",
"data": [
{
"name": "team_id",
"value": null
},
{
"name": "user_id",
"value": null
},
{
"name": "location_id",
"value": null
},
{
"name": "opponent_id",
"value": null
},
{
"name": "started_after",
"value": null
},
{
"name": "started_before",
"value": null
},
{
"name": "repeating_uuid",
"value": null
},
{
"name": "id",
"value": null
},
{
"name": "is_game",
"value": null
},
{
"name": "updated_since",
"value": null
},
{
"name": "page_size",
"value": null,
"prompt": "The number of items to return for each page. Sending this parameter with the query will enable paging for the returned collection."
},
{
"name": "page_number",
"value": null,
"prompt": "The number of the page to be returned. This requires that paging be turned on by also providing the page_size parameter."
},
{
"name": "sort_start_date",
"value": null,
"prompt": "Sort the returned dataset based on the start_date field, valid values are 'asc' or 'desc'."
}
]
},
{
"rel": "search_games",
"href": "https://api.teamsnap.com/v3/events/search_games",
"data": [
{
"name": "team_id",
"value": null
},
{
"name": "page_size",
"value": null,
"prompt": "The number of items to return for each page. Sending this parameter with the query will enable paging for the returned collection."
},
{
"name": "page_number",
"value": null,
"prompt": "The number of the page to be returned. This requires that paging be turned on by also providing the page_size parameter."
}
]
},
{
"rel": "overview",
"href": "https://api.teamsnap.com/v3/events/overview",
"data": [
{
"name": "team_id",
"value": null
}
]
}
],
"commands": [
{
"rel": "send_availability_reminders",
"href": "https://api.teamsnap.com/v3/events/send_availability_reminders",
"prompt": "members_to_notify = [member_id, member_id]",
"data": [
{
"name": "id",
"value": null
},
{
"name": "members_to_notify",
"value": null
},
{
"name": "notify_team_as_member_id",
"value": null
}
]
},
{
"rel": "update_final_score",
"href": "https://api.teamsnap.com/v3/events/update_final_score",
"prompt": "Update the final score for an event",
"data": [
{
"name": "id",
"value": null
},
{
"name": "points_for_team",
"value": null
},
{
"name": "points_for_opponent",
"value": null
},
{
"name": "shootout_points_for_team",
"value": null
},
{
"name": "shootout_points_for_opponent",
"value": null
},
{
"name": "is_overtime",
"value": null
},
{
"name": "is_shootout",
"value": null
},
{
"name": "results",
"value": null
},
{
"name": "results_url",
"value": null
}
]
},
{
"rel": "bulk_create",
"href": "https://api.teamsnap.com/v3/events/bulk_create",
"prompt": "event_ids = [event_id, event_id]",
"data": [
{
"name": "templates",
"value": null
},
{
"name": "team_id",
"value": null
},
{
"name": "notify_team_as_member_id",
"value": null
},
{
"name": "notify_team",
"value": null
}
]
}
],
"page_info": {
"total_items": 0,
"page_size": 1,
"page_number": 1
}
}
}
{
"time":"2019-10-02T02:50:31.001347Z",
"status":200,
"metadata":null,
"error":null,
"data":{
"uuid":"person-3192a9a0-2d87-0137-24ba-00163eab79c2",
"user_id":4209826,
"updated_at":"2019-03-20T21:43:59.000000",
"last_name":"McCraw",
"is_age_hidden":null,
"inserted_at":"2019-03-20T21:43:59.000000",
"id":143944,
"gender":null,
"first_name":"Dustin",
"contact_infos":null,
"birthdate":null
}
}
{
"data":[
{
"account":{
"address":{
"city":null,
"country":"US",
"line1":null,
"line2":null,
"postal_code":null,
"state":null
},
"business_type":"non_profit",
"charges_enabled":true,
"company":{
"name":"Stripe Org",
"tax_id_provided":false
},
"country":"US",
"created_at":"2019-05-06T20:11:55Z",
"currency":"usd",
"details_submitted":false,
"is_setup_complete":false,
"payouts_enabled":false,
"requirements":{
"current_deadline":null,
"currently_due":[
"external_account"
],
"disabled_reason":"requirements.past_due",
"eventually_due":[
"company.tax_id",
"external_account"
],
"past_due":[
"external_account"
],
"pending_verification":[
]
},
"stripes__account_id":"acct_1EXDAEE99KWL2RR4",
"tos_accept_date":"2019-02-14T15:21:22Z"
},
"bank_account":null,
"id":"148",
"is_setup_complete":false,
"person":null,
"resource_type":"payment_account",
"stripe_account_id":"acct_1EXDAEE99KWL2RR4",
"teamsnap_stripe_account_id":"148"
}
],
"meta":{
"code":"ok",
"status_code":200,
"timestamp":"2019-10-02T02:53:04.348694Z",
"version":"1.0.0",
"has_more":false
}
}
Challenge #3
What JSON format should our clients understand?
- Our services are returning different JSON formats.
- SNAPI, Accounts-api, Payments-Api, Hoyle
- All different formats!!!
- Every time a client accesses a new service we fire up, they must learn that service's specific json format.
- Because these are non-standard formats we can't use open source libraries for backend or clients.
Challenge #4
https://api.teamsnap.com/v3/team_preferences/search
https://api.teamsnap.com/v3/plans/search
https://api.teamsnap.com/v3/sports/search
https://api.teamsnap.com/v3/teams/search
https://api.teamsnap.com/v3/events/search
https://api.teamsnap.com/v3/locations/search
https://api.teamsnap.com/v3/opponents/search
https://api.teamsnap.com/v3/assignments/search
https://api.teamsnap.com/v3/availabilities/search
https://api.teamsnap.com/v3/event_lineup/search
https://api.teamsnap.com/v3/sport_positions/search
Challenge #4
What mobile has to download to display a single event.
https://api.teamsnap.com/v3/team_preferences/search
https://api.teamsnap.com/v3/plans/search
https://api.teamsnap.com/v3/sports/search
https://api.teamsnap.com/v3/teams/search
https://api.teamsnap.com/v3/events/search
https://api.teamsnap.com/v3/locations/search
https://api.teamsnap.com/v3/opponents/search
https://api.teamsnap.com/v3/assignments/search
https://api.teamsnap.com/v3/availabilities/search
https://api.teamsnap.com/v3/event_lineup/search
https://api.teamsnap.com/v3/sport_positions/search
Challenge #4
https://api.teamsnap.com/v3/broadcast_emails/search
https://api.teamsnap.com/v3/teams/search
https://api.teamsnap.com/v3/members/search
https://api.teamsnap.com/v3/member_email_addresses/search
https://api.teamsnap.com/v3/contacts/search
https://api.teamsnap.com/v3/contact_email_addresses/search
https://api.teamsnap.com/v3/division_member_email_addresses
/search
https://api.teamsnap.com/v3/division_contact_email_addresses
/search
Challenge #4
https://api.teamsnap.com/v3/broadcast_emails/search
https://api.teamsnap.com/v3/teams/search
https://api.teamsnap.com/v3/members/search
https://api.teamsnap.com/v3/member_email_addresses/search
https://api.teamsnap.com/v3/contacts/search
https://api.teamsnap.com/v3/contact_email_addresses/search
https://api.teamsnap.com/v3/division_member_email_addresses
/search
https://api.teamsnap.com/v3/division_contact_email_addresses
/search
What McFeely has to download to send a team email.
Challenge #4
How many endpoints do clients need to download to get all of our interconnected data?
- How much time are we adding to every render for all the multiple round trips with the latency for each call?
- Clients have to store all of this interconnected data
- Clients then stitch it back together
Challenge #5
Challenge #5
/schemas 49142 lines 661,485 bytes
Challenge #5
How do we document our apis for both internal and external clients?
- How do we document all of our services and make that data discoverable?
5 Challenges
- Multiple services/apis
- Large Collection+JSON payloads
- Custom JSON payload
- Multiple server/api round trips
- Api/Service Documentation
GraphQL Api Gateway
Let's wrap all of our services behind a GraphQL server which serves as an api gateway giving clients one documented entry point into all our our data.
What is an api gateway?
- An API gateway is programming that sits in front of an application programming interface (API) and acts as a single point of entry for a defined group of microservices.
What is an GraphQL?
GraphQL is a query language for APIs and a runtime for fulfilling those queries with your existing data.
GraphQL
- GraphQL is a query language created at Facebook in 2012
- Open source way to define an API
- Standard way to provide the description of the data of your api and it lets clients request exactly what they need
- Came from the conflict between users' wanting to get all the information in one query vs the requirement to isolated REST calls
GraphQL
Title Text
- Defines a data shape
- Hierarchical/Graph
- Strongly typed
- Protocol
- Introspective
Defines a Data Shape
The server defines a schema.
Clients compose queries against that schema and the response mirrors that query.
This allow clients determine what data they need and query for it.
Hierarchical/Graph/Relational
GraphQL defines the relationships between the objects in the schema.
Clients can choose what relationships they need for their specific use case.
Strongly Typed
Each level of a query corresponds to a type.
Each type has a set of available fields.
Define a Schema using scalar types.
i.e. A Character has a non null name and
a non null list of Episodes.
Schema takes care of all validation.
Protocol
Open source standard between server and clients.
Can be written in any backend language
using any datastore.
Each GraphQL field has its own resolver so a single GraphQL object can come from multiple services.
It's not a technology, it's a standard.
Introspective
A GraphQL server can be queried for the types it supports.
This creates a powerful platform for client software to build atop this information like code generation in statically typed languages
GraphQL
Advantages
- Retrieve only the data you need on consumer side
- Reduce the data volume returned by the server because you retrieve only what you need
- Reduce the number of calls to retrieve data by linking resources and aggregating queries
- Discover the schema you are querying
5 Challenges
- Multiple services/apis
- Large Collection+JSON payloads
- Custom JSON payload
- Multiple server/api round trips
- Api/Service Documentation
Challenge #1
Multiple services/api
By wrapping all of our services inside GraphQL, every client that interacts with TeamSnap only has to go to one endpoint to get all of our data.
Challenge #1
Multiple services/api
Alternatives:
A restful API gateway - This still has all the downsides of REST like multiple trips from the client
Challenge #2
Large Collection+JSON payloads
Since graphql responses mirror the query the clients send, it's almost the smallest payload possible.
Challenge #2
Large Collection+JSON payloads
Alternatives:
Use a smaller json format - This still has all the downsides of REST like multiple trips from the client. Doesn't allow clients to select the fields they want.
Challenge #3
Custom JSON payload
GraphQL is an open source standard. Every client/server framework conforms to the same format.
This allows us to use a lot of existing open source software like Absinthe and ApolloJS.
This allows amazing tooling like GraphQL Playground.
Challenge #3
Custom JSON payload
Alternatives:
Convert all of our services/api to use the same json format or JSON-Api - This still has all the downsides of REST like multiple trips from the client. Doesn't have the ability for clients to select the data they need.
Challenge #4
Multiple server/api round trips
The relationships between data allows client to pull related data together in a single query. Clients can even query completely different data that isn't related by just having multiple queries in the request.
Challenge #4
Alternatives:
Bulk load endpoint - Few round trips but doesn't have the ability for clients to select the data they need so the payload is still larger than it needs to be.
Stripe like expanding fields - Fewer round trips but creates a custom solution for expanding fields.
Multiple server/api round trips
Challenge #5
Api/Service Documentation
The Absinthe GraphQL framework allows for co-location of documentation and code so as you write code you write documentation.
Documentation is highly encouraged because it can easily be introspected by tools like GraphQL Playground.
Challenge #5
Alternatives:
Swagger - requires a lot of custom code to generate those documents
Api/Service Documentation
Want to learn more?
-
#graphql channel
-
https://github.com/teamsnap/fore_api_gateway
-
I am always free to talk graphql!
Thank you!
TeamSnap and GraphQl
By Dustin McCraw
TeamSnap and GraphQl
- 700