-
-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathgrouping.coffee
203 lines (165 loc) · 6.34 KB
/
grouping.coffee
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
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
###
SERVER METHODS
Hook in group id to all operations, including find
Grouping contains _id: userId and groupId: groupId
###
Partitioner = {}
Grouping = new Mongo.Collection("ts.grouping")
# Meteor environment variables for scoping group operations
Partitioner._currentGroup = new Meteor.EnvironmentVariable()
Partitioner._directOps = new Meteor.EnvironmentVariable()
###
Public API
###
Partitioner.setUserGroup = (userId, groupId) ->
check(userId, String)
check(groupId, String)
if Grouping.findOne(userId)
throw new Meteor.Error(403, "User is already in a group")
Grouping.upsert userId,
$set: {groupId: groupId}
Partitioner.getUserGroup = (userId) ->
check(userId, String)
Grouping.findOne(userId)?.groupId
Partitioner.clearUserGroup = (userId) ->
check(userId, String)
Grouping.remove(userId)
Partitioner.group = ->
# If group is overridden, return that instead
if (groupId = Partitioner._currentGroup.get())?
return groupId
try # We may be outside of a method
userId = Meteor.userId()
return unless userId
return Partitioner.getUserGroup(userId)
Partitioner.bindGroup = (groupId, func) ->
Partitioner._currentGroup.withValue(groupId, func);
Partitioner.bindUserGroup = (userId, func) ->
groupId = Partitioner.getUserGroup(userId)
unless groupId
Meteor._debug "Dropping operation because #{userId} is not in a group"
return
Partitioner.bindGroup(groupId, func)
Partitioner.directOperation = (func) ->
Partitioner._directOps.withValue(true, func);
# This can be replaced - currently not documented
Partitioner._isAdmin = (userId) -> Meteor.users.findOne(userId, {fields: groupId: 1, admin: 1}).admin is true
getPartitionedIndex = (index) ->
defaultIndex = { _groupId : 1 }
return defaultIndex unless index
return _.extend( defaultIndex, index )
Partitioner.partitionCollection = (collection, options) ->
# Because of the deny below, need to create an allow validator
# on an insecure collection if there isn't one already
if collection._isInsecure()
collection.allow
insert: -> true
update: -> true
remove: -> true
# Idiot-proof the collection against admin users
collection.deny
insert: Partitioner._isAdmin
update: Partitioner._isAdmin
remove: Partitioner._isAdmin
collection.before.find findHook
collection.before.findOne findHook
# These will hook the _validated methods as well
collection.before.insert insertHook
###
No update/remove hook necessary, see
https://github.com/matb33/meteor-collection-hooks/issues/23
###
# Index the collections by groupId on the server for faster lookups across groups
collection._ensureIndex getPartitionedIndex(options?.index), options?.indexOptions
# Publish admin and group for users that have it
Meteor.publish null, ->
return unless @userId
return Meteor.users.find @userId,
fields: {
admin: 1
group: 1
}
# Special hook for Meteor.users to scope for each group
userFindHook = (userId, selector, options) ->
return true if Partitioner._directOps.get() is true
return true if Helpers.isDirectUserSelector(selector)
groupId = Partitioner._currentGroup.get()
# This hook doesn't run if we're not in a method invocation or publish
# function, and Partitioner._currentGroup is not set
return true if (!userId and !groupId)
unless groupId
user = Meteor.users.findOne(userId, {fields: groupId: 1, admin: 1})
groupId = Grouping.findOne(userId)?.groupId
# If user is admin and not in a group, proceed as normal (select all users)
return true if user.admin and !groupId
# Normal users need to be in a group
throw new Meteor.Error(403, ErrMsg.groupErr) unless groupId
# Since user is in a group, scope the find to the group
filter =
"group" : groupId
"admin": {$exists: false}
unless @args[0]
@args[0] = filter
else
_.extend(selector, filter)
return true
# Attach the find hooks to Meteor.users
Meteor.users.before.find userFindHook
Meteor.users.before.findOne userFindHook
# No allow/deny for find so we make our own checks
findHook = (userId, selector, options) ->
# Don't scope for direct operations
return true if Partitioner._directOps.get() is true
# for find(id) we should not touch this
# TODO this may allow arbitrary finds across groups with the right _id
# We could amend this in the future to {_id: someId, _groupId: groupId}
# https://github.com/mizzao/meteor-partitioner/issues/9
# https://github.com/mizzao/meteor-partitioner/issues/10
return true if Helpers.isDirectSelector(selector)
if userId
# Check for global hook
groupId = Partitioner._currentGroup.get()
unless groupId
throw new Meteor.Error(403, ErrMsg.userIdErr) unless userId
groupId = Grouping.findOne(userId)?.groupId
throw new Meteor.Error(403, ErrMsg.groupErr) unless groupId
# if object (or empty) selector, just filter by group
unless selector?
@args[0] = { _groupId : groupId }
else
selector._groupId = groupId
# Adjust options to not return _groupId
unless options?
@args[1] = { fields: {_groupId: 0} }
else
# If options already exist, add {_groupId: 0} unless fields has {foo: 1} somewhere
options.fields ?= {}
options.fields._groupId = 0 unless _.any(options.fields, (v) -> v is 1)
return true
insertHook = (userId, doc) ->
# Don't add group for direct inserts
return true if Partitioner._directOps.get() is true
groupId = Partitioner._currentGroup.get()
unless groupId
throw new Meteor.Error(403, ErrMsg.userIdErr) unless userId
groupId = Grouping.findOne(userId)?.groupId
throw new Meteor.Error(403, ErrMsg.groupErr) unless groupId
doc._groupId = groupId
return true
# Sync grouping needed for hooking Meteor.users
Grouping.find().observeChanges
added: (id, fields) ->
unless Meteor.users.update(id, $set: {"group": fields.groupId} )
Meteor._debug "Tried to set group for nonexistent user #{id}"
return
changed: (id, fields) ->
unless Meteor.users.update(id, $set: {"group": fields.groupId} )
Meteor._debug "Tried to change group for nonexistent user #{id}"
removed: (id) ->
unless Meteor.users.update(id, $unset: {"group": null} )
Meteor._debug "Tried to unset group for nonexistent user #{id}"
TestFuncs =
getPartitionedIndex: getPartitionedIndex
userFindHook: userFindHook
findHook: findHook
insertHook: insertHook