by Gerard Sans | @gerardsans
data:image/s3,"s3://crabby-images/8c44f/8c44f70d816f9598ddeebdcbe099049d8a88c0f0" alt=""
Not a basic offline first app using
GraphQL and
Amplify DataStore
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=""
Not a basic offline first app using
GraphQL and
Amplify DataStore
data:image/s3,"s3://crabby-images/f76be/f76be5b3fdd688dd14cdc1e76c0646632f22f0c2" alt=""
data:image/s3,"s3://crabby-images/d2f57/d2f5768f79efaf12608b25366aab12e0c36061f1" alt=""
data:image/s3,"s3://crabby-images/5f13e/5f13e3dcb2985c57b4e6252cd73f7812cde4e016" 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 152 events in 37 countries
What is Offline first?
data:image/s3,"s3://crabby-images/ef33c/ef33cd8a04386b9c4019b10ba3f516a423a89bcc" alt=""
Offline First for Web
Reliable
Storage
Native-like Features
Great User Experience
Offline
Ready
Progressive Web Apps
Reach
Capabilities
Native
Applications
PWA
Applications
Web/SPA
Applications
PWA Core Requirements
Web App
Manifest
Service
Worker
Security
(HTTPS)
Service Worker Online
index.html
Cache
Hosting
Service
Worker
app.js
app.js
app.js
logo.png
logo.png
logo.png
app.js
logo.png
app.js
logo.png
Service Worker Offline
index.html
Cache
Hosting
Service
Worker
OFFLINE
app.js
logo.png
app.js
logo.png
app.js
logo.png
app.js
logo.png
data:image/s3,"s3://crabby-images/0f0c4/0f0c449f06322de6987c4c9ebf5a5cc4cb701b01" alt=""
AWS AMPLIFY
data:image/s3,"s3://crabby-images/6682c/6682cd40dc5a77d882c8855d5f5766ad4e87fe4c" alt=""
Fullstack Serverless
🦄
🌩️
data:image/s3,"s3://crabby-images/00625/006250d6fd6711c1875b8c1b1824f4e876cb1965" alt=""
Update
Init
Add
Push
Amplify CLI
data:image/s3,"s3://crabby-images/6682c/6682cd40dc5a77d882c8855d5f5766ad4e87fe4c" alt=""
Categories
interactions
storage
notifications
auth
analytics
function
api
hosting
xr
transcribe
translate
polly
rekognition
comprehend
data:image/s3,"s3://crabby-images/20b14/20b14f33ec67844851cb9b2f9631b5439fdc0a65" alt=""
Chatty
data:image/s3,"s3://crabby-images/f76be/f76be5b3fdd688dd14cdc1e76c0646632f22f0c2" alt=""
data:image/s3,"s3://crabby-images/d3d8a/d3d8a7f6b2f0ecde7e88c30748b26321d4927374" alt=""
Chatty App
data:image/s3,"s3://crabby-images/c7bda/c7bda0f2ddb93819a42811adb7e8983b96243e61" alt=""
data:image/s3,"s3://crabby-images/c7bda/c7bda0f2ddb93819a42811adb7e8983b96243e61" alt=""
AppSync
data:image/s3,"s3://crabby-images/c7bda/c7bda0f2ddb93819a42811adb7e8983b96243e61" alt=""
GraphQL
$ amplify init
$ amplify add auth
$ amplify add api
$ amplify push
src/app.component.ts
Amplify CLI commands
type Chatty @model {
id: ID!
user: String!
message: String!
createdAt: AWSDateTime
}
src/app.component.ts
Chatty GraphQL Schema
Amplify DataStore Offline
data:image/s3,"s3://crabby-images/14938/149386bbd7f31b0b815a33309efb1ece15c61603" alt=""
Amplify DataStore Online
data:image/s3,"s3://crabby-images/14938/149386bbd7f31b0b815a33309efb1ece15c61603" alt=""
data:image/s3,"s3://crabby-images/14938/149386bbd7f31b0b815a33309efb1ece15c61603" alt=""
import { DataStore } from "@aws-amplify/datastore";
import { Chatty } from "./models";
await DataStore.save(new Chatty({
user: "gsans",
message: "Hi everyone!👋",
createdAt: new Date().toISOString()
}))
src/app.component.ts
Creating a Message
import { DataStore, Predicates } from "@aws-amplify/datastore";
import { Chatty } from "./models";
const msg = await DataStore.query(Chatty, Predicates.ALL);
src/app.component.ts
Querying data
Make it
a PWA!
$ vue add @vue/pwa
src/main.js (Updated)
> + import './registerServiceWorker'
src/registerServiceWorker.js (Added)
src/app.component.ts
Run PWA Vue CLI plugin
$ yarn build
src/app.component.ts
Test your PWA
└── dist
├── css
│ └── app.<version>.css
├── img/icons
│ ├── android-chrome-<size>.png
│ └── ...
├── js
│ ├── app.<version>.png
│ └── ...
├── favicon.ico
├── index.html
├── manifest.json
├── precache-manifest.<version>.json
├── robots.txt
└── service-worker.js
$ cd dist
/dist$ python -m SimpleHTTPServer 8887
data:image/s3,"s3://crabby-images/83948/83948a7dd708d2dd80c9d1185a1499aec0ce6524" alt=""
// vue.config.js
const manifest = require('./public/manifest.json')
module.exports = {
pwa: {
name: manifest.short_name,
themeColor: manifest.theme_color,
msTileColor: manifest.background_color,
appleMobileWebAppCapable: 'yes',
appleMobileWebAppStatusBarStyle: 'black',
workboxPluginMode: 'InjectManifest',
workboxOptions: {
swSrc: 'src/service-worker.js',
}
src/app.component.ts
Add custom configuration (1/2)
// src/service-worker.js
workbox.core.setCacheNameDetails({ prefix: 'amplify-datastore' })
workbox.core.skipWaiting()
workbox.core.clientsClaim()
const cacheFiles = [{
"revision": "e653ab4d124bf16b5232",
"url": "https://aws-amplify.github.io/img/amplify.svg"
}]
self.__precacheManifest = cacheFiles.concat(self.__precacheManifest || [])
workbox.precaching.precacheAndRoute(self.__precacheManifest, {})
src/app.component.ts
Add custom configuration (2/2)
Chatty
PWA
data:image/s3,"s3://crabby-images/569cf/569cf2119d80af0fc00b8f08399ffd6e06ee0e4a" alt=""
Web, Mobile and Desktop
data:image/s3,"s3://crabby-images/cc9ef/cc9efb82b46e207f59ca8eaa924fee2f90441586" alt=""
data:image/s3,"s3://crabby-images/cc9ef/cc9efb82b46e207f59ca8eaa924fee2f90441586" alt=""
data:image/s3,"s3://crabby-images/cc9ef/cc9efb82b46e207f59ca8eaa924fee2f90441586" alt=""
Add to home screen
data:image/s3,"s3://crabby-images/628d4/628d41c41fcf3cc08f58abff6ef76d8e52af4370" alt=""
data:image/s3,"s3://crabby-images/a4f73/a4f73bc9945a84c38407e0dd777b294eeb285c5f" alt=""
data:image/s3,"s3://crabby-images/0c8cc/0c8cc882c161b0b09d2a54d15a677d0a0a729efa" alt=""
data:image/s3,"s3://crabby-images/7d90f/7d90f41eb0efc16b7275cc4d76cbd0f55ede45ab" alt=""
Passes Lighthouse Test
data:image/s3,"s3://crabby-images/51241/51241462aa484312a9d038df6cbf85e7ece222f5" alt=""
99% Offline-ready
- Survives offline reload
- Stores messages while offline
- Shares messages when back online
- User doesn't know if App is offline
// <div v-if="offline">You are offline.</div>
// App.vue
import { Hub } from 'aws-amplify';
export default {
data() {
return { offline: undefined };
},
created() {
this.listener = Hub.listen('datastore', {payload:{event}} => {
if (event === 'networkStatus') {
this.offline = !data.active;
}
})
}
}
src/app.component.ts
Improve Offline UX
data:image/s3,"s3://crabby-images/b638c/b638cc1dfcbb25dc173404d0e12964c2b838acb2" alt=""
- Faster (pre-cached)
- Online/Offline
- Full screen app
- Browser/Mobile/Desktop
- Web/Native-like features
- Slower
- Online
- Tab in browser
- Browser
- Web features
Web/SPA vs PWA
data:image/s3,"s3://crabby-images/8cf3f/8cf3f14d3e251e44c5962201ba0272c4c4736a88" alt=""
Try it!
amplify-datastore-chatty-pwa-vue (solution)
amplify-datastore-chatty-vue (workshop)
data:image/s3,"s3://crabby-images/d2f57/d2f5768f79efaf12608b25366aab12e0c36061f1" alt=""
data:image/s3,"s3://crabby-images/d2f57/d2f5768f79efaf12608b25366aab12e0c36061f1" alt=""
data:image/s3,"s3://crabby-images/d2f57/d2f5768f79efaf12608b25366aab12e0c36061f1" alt=""
Not a basic offline-first app using GraphQL and Amplify DataStore
By Gerard Sans
Not a basic offline-first app using GraphQL and Amplify DataStore
Offline-first apps need to support: intermittent connectivity, transition seamlessly between online and offline states, reliable CRUD on-device data, data synchronisation and data conflict resolution to enable real-time collaboration. Amplify DataStore is an on device persistent repository for interacting with local data and able to automatically synchronize via GraphQL. Using Amplify DataStore will allow us to implement offline-first while using a simple programming model.
- 2,848