This repository was archived by the owner on May 30, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathdynamodb_big_segment_store.js
93 lines (82 loc) · 2.88 KB
/
dynamodb_big_segment_store.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
const { initState } = require('./dynamodb_helpers');
const { promisify } = require('util');
const keyMetadata = 'big_segments_metadata';
const keyUserData = 'big_segments_user';
const attrSyncTime = 'synchronizedOn';
const attrIncluded = 'included';
const attrExcluded = 'excluded';
// Note that the format of parameters in this implementation is a bit different than in the
// LD DynamoDB integrations for some other platforms, because we are using the
// AWS.DynamoDB.DocumentClient class, which represents values as simple types like
// string or number, rather than in the { S: stringValue } or { N: numericStringValue }
// format used by the basic AWS DynamoDB API.
function DynamoDBBigSegmentStore(tableName, maybeOptions) {
const options = maybeOptions || {};
return () => // config parameter is currently unused because we don't need to do any logging
dynamoDBBigSegmentStoreImpl(tableName, options);
}
function dynamoDBBigSegmentStoreImpl(tableName, options) {
const state = initState(options);
const dynamoDBClient = state.client;
const prefix = state.prefix;
const store = {};
// Pre-promisify for efficiency. Note that we have to add .bind(client) to each method when
// when using promisify, because the AWS client methods don't work without a "this" context.
const clientGet = promisify(dynamoDBClient.get.bind(dynamoDBClient));
store.getMetadata = async () => {
const key = prefix + keyMetadata;
const data = await clientGet({
TableName: tableName,
Key: { namespace: key, key: key },
});
if (data.Item) {
const attr = data.Item[attrSyncTime];
if (attr) {
return { lastUpToDate: attr };
}
}
return { lastUpToDate: undefined };
};
store.getUserMembership = async userHashKey => {
const data = await clientGet({
TableName: tableName,
Key: {
namespace: prefix + keyUserData,
key: userHashKey,
},
});
const item = data.Item;
if (item) {
const membership = {};
const excludedRefs = item[attrExcluded];
const includedRefs = item[attrIncluded];
// The actual type of these values in DynamoDB is a string set. The DocumentClient
// returns string set values as a special type where the actual list of strings is
// in a "values" property.
if (excludedRefs && excludedRefs.values) {
for (const ref of excludedRefs.values) {
membership[ref] = false;
}
}
if (includedRefs && includedRefs.values) {
for (const ref of includedRefs.values) {
membership[ref] = true;
}
}
return membership;
}
return null;
};
store.close = function() {
// The Node DynamoDB client is stateless, so close isn't a meaningful operation.
};
return store;
}
module.exports = {
DynamoDBBigSegmentStore,
keyMetadata,
keyUserData,
attrSyncTime,
attrIncluded,
attrExcluded,
};