-
Notifications
You must be signed in to change notification settings - Fork 1
/
03 - Batching.js
161 lines (149 loc) · 4.37 KB
/
03 - Batching.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
/**
* Lesson 03 - Batching
*
* Batching is a second layer of abstraction
* where instead of resolving requests immediately, we pause for a bit
*
* And then we take every ID that was requested over that pause
* and do one single query for all of them
*
* Lets call that "activeQuery" and make it null by default.
*
* To pause, we need to make a new Promise with a short timeout
* 1 millisecond is ok here. Longer will allow bigger batches
* but it will also make the requests slower.
*
* Our batch size will be every ID in the promises cache, which
* gets us everything requested so far
*
* Bring the database query code here, and we'll tweak it to accept
* a list of IDs instead of just a single one
*
* That's going to be Select all from users where id in "ids"
*
* .where becomes .whereIn, and pass it all the IDs from the promiseCache
*
* We don't just want the first item anymore, so we can remove .first()
* and this becomes the activeQuery instead
*
* If we're fetching the list of IDs from the cache, then we're going to
* be missing the current one if it's not a cache hit. Lots of ways to
* fix that, I'll just add it to the cache and be done with it
*
* So the promise for each item key is going to start with the common
* activeQuery (which returns every item in this batch) and then branch
* off to find and return the specific ID for this item
*
* Now we resolve the promise for this key, which is being returned
* from the load method for this key.
*
* Last thing – once this batch has started, we don't need to dispatch it again
* so only do that if we haven't set activeQuery yet
*
* Lets test this out
*
query {
posts {
id
title
author {
id
name
}
}
}
*
* One request for posts, and one request for all of the users at once. Before,
* 1000 posts would do 1000 individual author requests. Now it will be 1 request
* no matter how many posts we have
*
* Running this a second time shows our cache still works across multiple network requests
*
* Since there's no guarantee those requests come from the same person,
* That means a logged-in user could cache some sensitive data that guest
* users could access by mistake
*/
const { ApolloServer, gql } = require('apollo-server')
const sql = require('knex')({
client: 'pg',
connection: {
host : '127.0.0.1',
port : 5432,
user : 'postgres',
password : 'password',
database : 'postgres'
}
});
// A schema is a collection of type definitions (hence "typeDefs")
// that together define the "shape" of queries that are executed against
// your data.
const typeDefs = gql`
type Query {
posts: [Post]
}
type Post {
id: String
title: String
author: User
}
type User {
id: String
name: String
}
`
const cache = {
promises: {},
activeQuery: null,
load(key) {
if (this.promises[key]) {
// We've queried this before, return the cached promise
return this.promises[key]
} else {
// Just a placeholder until the batch is ready to go
this.promises[key] = null
}
return new Promise((resolve) => {
// Wait until the batch is ready
// Increase the timeout to make larger, slower batches
setTimeout(() => {
if (!this.activeQuery) {
// Query all IDs at once
const ids = Object.keys(this.promises)
console.log(`SELECT * from users WHERE id in (${ids.join(',')})`)
this.activeQuery = sql
.select('*')
.from('users')
.whereIn('id', ids)
}
// Cache a promise that waits on the active query
this.promises[
key
] = this.activeQuery.then((items) => {
// And selects the item with this id
return items.find((item) => item.id === Number(key))
})
resolve(this.promises[key])
}, 1)
})
},
}
const resolvers = {
Query: {
posts() {
// Executes once per query
console.log('SELECT * from posts')
return sql('posts').select('*')
},
},
Post: {
async author(post) {
// Executes once per post per query
return cache.load(post.author_id)
},
},
}
const server = new ApolloServer({ typeDefs, resolvers })
// The `listen` method launches a web server.
server.listen().then(({ url }) => {
console.log(`🚀 Server ready at ${url}`);
})