by Gerard Sans | @gerardsans
data:image/s3,"s3://crabby-images/8c44f/8c44f70d816f9598ddeebdcbe099049d8a88c0f0" alt=""
Building a
serverless geolocation API
data:image/s3,"s3://crabby-images/f6e54/f6e544ddb04ae4ed79746794b47a77180f3d9300" alt=""
data:image/s3,"s3://crabby-images/f6e54/f6e544ddb04ae4ed79746794b47a77180f3d9300" alt=""
data:image/s3,"s3://crabby-images/f6e54/f6e544ddb04ae4ed79746794b47a77180f3d9300" alt=""
Building a
serverless geolocation API
data:image/s3,"s3://crabby-images/354a7/354a7ba6b2dd8c91b3d3942062f8305f56c92a81" alt=""
SANS
GERARD
Developer Advocate AWS
data:image/s3,"s3://crabby-images/fe78c/fe78ce96cdeb1c682e2b123d8f48014acbde375d" alt=""
Developer Advocate AWS
data:image/s3,"s3://crabby-images/5b095/5b095dded2bf443b9f8cb641e33af9921956c2db" alt=""
International Speaker
data:image/s3,"s3://crabby-images/07d8b/07d8b538ae01b1fdef8e1a86d9d804eb05f3565e" alt=""
Spoken at 143 events in 37 countries
Bike Sharing
data:image/s3,"s3://crabby-images/de263/de263ec5379aa30e96b95273d1756713bcfe46fc" alt=""
data:image/s3,"s3://crabby-images/61d04/61d047c57df3e69c43636b5434a92e95aba81bb0" alt=""
data:image/s3,"s3://crabby-images/ca43b/ca43b6911edb6054e82f4861668c19a0841d72cf" alt=""
data:image/s3,"s3://crabby-images/d7cd5/d7cd5d995aee22d507e48aaff6ea65ece34d3413" alt=""
Benefits
- Transport flexibility
- Reduce traffic emissions
- Reduce traffic congestion
- Health benefits for users
Santander Cycles
data:image/s3,"s3://crabby-images/7f83f/7f83f39416c55303f483223dd2c9e9969b2387ae" alt=""
Overview
- Since 2010
- +12,000 Bikes
- 778 stations
Boris Bikes
LNDBikes
Fullstack Serverless
🦄
🌩️
No servers to manage
Fault tolerance High availability
Never pay for idle usage
Auto-scales immediately
Serverless
$
AWS AMPLIFY
data:image/s3,"s3://crabby-images/6682c/6682cd40dc5a77d882c8855d5f5766ad4e87fe4c" alt=""
data:image/s3,"s3://crabby-images/00625/006250d6fd6711c1875b8c1b1824f4e876cb1965" alt=""
Categories
interactions
storage
notifications
auth
analytics
function
amplify add <category>
api
hosting
xr
transcribe
rekognition
translate
comprehend
amplify add predictions
polly
data:image/s3,"s3://crabby-images/e153a/e153af896f6fb27e953249861cc37fabe3a6986a" alt=""
data:image/s3,"s3://crabby-images/0359f/0359f0b0429b5e32968b74838c4470bb2e3de3b2" alt=""
data:image/s3,"s3://crabby-images/e6a0b/e6a0b2069d0db739c7f377075a49bd31c3949fae" alt=""
data:image/s3,"s3://crabby-images/79ad5/79ad5414a5b369b1f4c4e5e91db6fb6df205f5b3" alt=""
data:image/s3,"s3://crabby-images/dcb75/dcb751dfd6d12d4d971735dc0472d6887708ae32" alt=""
data:image/s3,"s3://crabby-images/20b14/20b14f33ec67844851cb9b2f9631b5439fdc0a65" alt=""
London
Unified API
/BikePoint
/BikePoint/id
TfL Unified API
/BikePoint/Search
[
{
"id": "BikePoints_1",
"commonName": "River Street , Clerkenwell",
"additionalProperties": [{
"key": "NbBikes", "value": "11",
}],
"lat": 51.529163,
"lon": -0.10997
}
// 777 more
]
/BikePoint
{
"id": "BikePoints_1",
"commonName": "River Street , Clerkenwell",
"additionalProperties": [{
"key": "NbBikes", "value": "11",
}],
"lat": 51.529163,
"lon": -0.10997
}
/BikePoint/BikePoints_1
data:image/s3,"s3://crabby-images/0f0c4/0f0c449f06322de6987c4c9ebf5a5cc4cb701b01" alt=""
Loading bike stations
Data Transformations
GeoJSONfeature
BikesPoint
{
"type": "Feature",
"geometry": {
"type": "Point",
"coordinates": [-0.10997, 51.529163]
},
"properties": {
"id": "BikePoints_1",
"name": "River Street , Clerkenwell"
}
}
geoJSON feature
data:image/s3,"s3://crabby-images/3fda4/3fda4ab16adaa79793fb196131677632104d16ff" alt=""
Coordinates = [-0.10997, 51.529163]
51.529163
-0.10997
Coordinates
data:image/s3,"s3://crabby-images/4432e/4432e43879bc48edfd6739bcbf595a7cdc9b47df" alt=""
Data Transformations
GeoJSONfeature
BikesPoint
mapbox Source
mapbox Layer
REST API integration
type BikePoint @model {
id: ID!
name: String!
description: String
location: Location
bikes: Int
}
GraphQL Schema
// request VTL template
{
"version": "2018-05-29",
"method": "GET",
"resourcePath": "/BikePoint/$context.source.id",
}
BikePoint.bikes HTTP Resolver
// response VTL template
#set($body = $util.parseJson($ctx.result.body))
#if($ctx.error)
$util.error($ctx.error.message, $ctx.error.type)
#end
#if($ctx.result.statusCode == 200)
$body.additionalProperties[6].value
#else
#return
#end
BikePoint.bikes HTTP Resolver
Adding Search
type BikePoint @model @searchable {
id: ID!
name: String!
description: String
location: Location
bikes: Int
}
type Location {
lat: Float
lon: Float
}
type Query {
nearbyBikeStations(location: LocationInput!, m: Int, limit: Int)
}
GraphQL Schema
Data Transformations
GeoJSONfeature
BikesPoint
GraphQL API
Elastic Search
# Create index
PUT /bikepoint
# Setup location type as geo_point
PUT /bikepoint/_mapping/doc
{
"properties": {
"location": {
"type": "geo_point"
}
}
}
Elastic Search index
mutation addBikePoint {
createBikePoint(input: {
id: "BikePoints_1"
name: "River Street , Clerkenwell"
location: {
lat: 51.529163
lon: -0.10997
}
}) { id }
}
Automatic indexing
GET /bikepoint/doc/_search
{
"query": {
"bool" : {
"must" : { "match_all" : {} },
"filter" : {
"geo_distance" : {
"distance" : "500m",
"distance_type": "arc",
"location" : {
"lon": -0.134167, "lat": 51.510239
}
Query nearbyBikeStations (1/2)
"sort": [{
"_geo_distance": {
"location": {
"lon": -0.134167, "lat": 51.510239
},
"order": "asc",
"unit": "m",
"distance_type": "arc"
}
}]
}
Query nearbyBikeStations (2/2)
data:image/s3,"s3://crabby-images/73de4/73de45660c1cca201198f942e63b8fbcca39907d" alt=""
Distance Calculations
data:image/s3,"s3://crabby-images/0bf6f/0bf6f106ef0d56e523291ed0baa41539254631a4" alt=""
data:image/s3,"s3://crabby-images/75fea/75fea681c03ffa1fdc5bede1fbf222a2b20ba344" alt=""
[lon1, lat1]
[lon2, lat2]
Haversine Formula
Great-circle distance
Distance Calculations
double distance(double lat1, double lon1, double lat2, double lon2) {
return 6378137 * haversine(lat1, lon1, lat2, lon2);
}
double haversine(double lat1, double lon1, double lat2, double lon2) {
double hsinX = Math.sin((lon1 - lon2) * 0.5);
double hsinY = Math.sin((lat1 - lat2) * 0.5);
double h = hsinY * hsinY + (Math.cos(lat1) * Math.cos(lat2) * hsinX * hsinX);
return 2 * Math.atan2(Math.sqrt(h), Math.sqrt(1 - h));
}
ElasticSearch arc distance (WGS 84)
data:image/s3,"s3://crabby-images/8cf3f/8cf3f14d3e251e44c5962201ba0272c4c4736a88" alt=""
More
data:image/s3,"s3://crabby-images/a69b3/a69b3e115be93b5720d9ecfabe6a42e9c0d43a23" alt=""
@undef_obj
@kurtiskemple
@dabit3
data:image/s3,"s3://crabby-images/5d85a/5d85a6bcaf318b657d00010f19e87391fc7b5afe" alt=""
data:image/s3,"s3://crabby-images/61c2f/61c2f4f9f4526d47dc3290c3d127fa2d0b15b8fe" alt=""
Kurt Kemple
Richardo
Nader Dabit
@TheSwaminator
data:image/s3,"s3://crabby-images/87d6a/87d6abaf0869992c369699322d5016cfe92c4764" alt=""
Nikhil Swaminathan
data:image/s3,"s3://crabby-images/354a7/354a7ba6b2dd8c91b3d3942062f8305f56c92a81" alt=""
data:image/s3,"s3://crabby-images/354a7/354a7ba6b2dd8c91b3d3942062f8305f56c92a81" alt=""
data:image/s3,"s3://crabby-images/354a7/354a7ba6b2dd8c91b3d3942062f8305f56c92a81" alt=""
Building a Serverless Geolocation Search API Using GraphQL
By Gerard Sans
Building a Serverless Geolocation Search API Using GraphQL
In this talk, we will be using AWS AppSync & Elasticsearch to create a geolocation search service API that will allow users to search within a distance from a given location and display results in a map. We will start designing our schema to introduce the geolocation data, implementing resolvers while discussing best practices and architecture decisions for the presented solution.
- 3,016