Skip to content

Commit f5555c0

Browse files
authoredNov 21, 2019
Added abitlity to make VFS group checking non-strict (#22) (#23)
The `strictGroups: false` mountpoint attribute can now make it so group matching is less strict. I.e. user can belong to *some* of the groups, not *all* of them.
1 parent bff477d commit f5555c0

File tree

3 files changed

+96
-36
lines changed

3 files changed

+96
-36
lines changed
 

‎__tests__/utils/vfs.js

+80-23
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,31 @@ const {Readable, Stream} = require('stream');
22
const temp = require('temp');
33
const utils = require('../../src/utils/vfs.js');
44

5+
const checkMountpointGroupPermission = (
6+
userGroups = [],
7+
mountpointGroups = [],
8+
strict
9+
) => {
10+
const check = utils.checkMountpointPermission({
11+
session: {
12+
user: {
13+
groups: userGroups
14+
}
15+
}
16+
}, {}, 'readdir', false, strict);
17+
18+
const mount = {
19+
name: 'osjs',
20+
root: 'osjs:/',
21+
attributes: {
22+
readOnly: true,
23+
groups: mountpointGroups
24+
}
25+
};
26+
27+
return check({mount});
28+
};
29+
530
describe('VFS Utils', () => {
631
afterAll(() => {
732
temp.cleanupSync();
@@ -42,21 +67,37 @@ describe('VFS Utils', () => {
4267
});
4368

4469
test('validateGroups - flat groups', () => {
45-
const mount = {
70+
expect(utils.validateGroups([
71+
'successful',
72+
'failure'
73+
], '', {
4674
attributes: {
4775
groups: [
4876
'successful'
4977
]
5078
}
51-
};
79+
})).toBe(true);
80+
5281
expect(utils.validateGroups([
53-
'successful',
5482
'failure'
55-
], '', mount)).toBe(true);
83+
], '', {
84+
attributes: {
85+
groups: [
86+
'successful'
87+
]
88+
}
89+
})).toBe(false);
5690

5791
expect(utils.validateGroups([
58-
'failure'
59-
], '', mount)).toBe(false);
92+
'successful'
93+
], '', {
94+
attributes: {
95+
groups: [
96+
'successful',
97+
'successful2'
98+
]
99+
}
100+
}, false)).toBe(true);
60101
});
61102

62103
test('validateGroups - method maps', () => {
@@ -99,27 +140,43 @@ describe('VFS Utils', () => {
99140
.toThrowError('Mountpoint \'osjs\' is read-only');
100141
});
101142

102-
test('checkMountpointPermission - groups', () => {
103-
const check = utils.checkMountpointPermission({
104-
session: {
105-
user: {
106-
groups: []
107-
}
108-
}
109-
}, {}, 'readdir', false);
143+
test('checkMountpointPermission - groups', async () => {
144+
await expect(checkMountpointGroupPermission(
145+
[],
146+
['required']
147+
))
148+
.rejects
149+
.toThrowError('Permission was denied for \'readdir\' in \'osjs\'');
110150

111-
const mount = {
112-
name: 'osjs',
113-
root: 'osjs:/',
114-
attributes: {
115-
readOnly: true,
116-
groups: ['required']
117-
}
118-
};
151+
await expect(checkMountpointGroupPermission(
152+
['missing'],
153+
['required']
154+
))
155+
.rejects
156+
.toThrowError('Permission was denied for \'readdir\' in \'osjs\'');
119157

120-
return expect(check({mount}))
158+
await expect(checkMountpointGroupPermission(
159+
['required'],
160+
['required', 'some-other']
161+
))
121162
.rejects
122163
.toThrowError('Permission was denied for \'readdir\' in \'osjs\'');
164+
165+
await expect(checkMountpointGroupPermission(
166+
['required'],
167+
['required']
168+
)).resolves.toBe(true);
169+
170+
await expect(checkMountpointGroupPermission(
171+
['required', 'some-other'],
172+
['required']
173+
)).resolves.toBe(true);
174+
175+
await expect(checkMountpointGroupPermission(
176+
['required'],
177+
['required', 'some-other'],
178+
false
179+
)).resolves.toBe(true);
123180
});
124181

125182
test('checkMountpointPermission', () => {

‎src/utils/vfs.js

+10-10
Original file line numberDiff line numberDiff line change
@@ -73,40 +73,40 @@ const streamFromRequest = req => {
7373
: fs.createReadStream(req.files.upload.path);
7474
};
7575

76-
const validateAll = (arr, compare) => arr.every(g => compare.indexOf(g) !== -1);
76+
const validateAll = (arr, compare, strict = true) => arr[strict ? 'every' : 'some'](g => compare.indexOf(g) !== -1);
7777

7878
/**
7979
* Validates array groups
8080
*/
81-
const validateNamedGroups = (groups, userGroups) => {
81+
const validateNamedGroups = (groups, userGroups, strict) => {
8282
const namedGroups = groups
8383
.filter(g => typeof g === 'string');
8484

8585
return namedGroups.length
86-
? validateAll(namedGroups, userGroups)
86+
? validateAll(namedGroups, userGroups, strict)
8787
: true;
8888
};
8989

9090
/**
9191
* Validates matp of groups based on method:[group,...]
9292
*/
93-
const validateMethodGroups = (groups, userGroups, method) => {
93+
const validateMethodGroups = (groups, userGroups, method, strict) => {
9494
const methodGroups = groups
9595
.find(g => typeof g === 'string' ? false : (method in g));
9696

9797
return methodGroups
98-
? validateAll(methodGroups[method], userGroups)
98+
? validateAll(methodGroups[method], userGroups, strict)
9999
: true;
100100
};
101101

102102
/**
103103
* Validates groups
104104
*/
105-
const validateGroups = (userGroups, method, mountpoint) => {
105+
const validateGroups = (userGroups, method, mountpoint, strict) => {
106106
const groups = mountpoint.attributes.groups || [];
107107
if (groups.length) {
108-
const namedValid = validateNamedGroups(groups, userGroups, method);
109-
const methodValid = validateMethodGroups(groups, userGroups, method);
108+
const namedValid = validateNamedGroups(groups, userGroups, strict);
109+
const methodValid = validateMethodGroups(groups, userGroups, method, strict);
110110

111111
return namedValid && methodValid;
112112
}
@@ -117,7 +117,7 @@ const validateGroups = (userGroups, method, mountpoint) => {
117117
/**
118118
* Checks permissions for given mountpoint
119119
*/
120-
const checkMountpointPermission = (req, res, method, readOnly) => {
120+
const checkMountpointPermission = (req, res, method, readOnly, strict) => {
121121
const userGroups = req.session.user.groups;
122122

123123
return ({mount}) => {
@@ -135,7 +135,7 @@ const checkMountpointPermission = (req, res, method, readOnly) => {
135135
}
136136
}
137137

138-
if (validateGroups(userGroups, method, mount)) {
138+
if (validateGroups(userGroups, method, mount, strict)) {
139139
return Promise.resolve(true);
140140
}
141141

‎src/vfs.js

+6-3
Original file line numberDiff line numberDiff line change
@@ -125,7 +125,8 @@ const createRequestFactory = findMountpoint => (getter, method, readOnly, respon
125125
}
126126
}
127127

128-
await checkMountpointPermission(req, res, method, readOnly)(found);
128+
const strict = found.mount.attributes.strictGroups !== false;
129+
await checkMountpointPermission(req, res, method, readOnly, strict)(found);
129130

130131
const result = await found.adapter[method]({req, res, adapter: found.adapter, mount: found.mount})(...args);
131132

@@ -145,8 +146,10 @@ const createCrossRequestFactory = findMountpoint => (getter, method, respond) =>
145146
const sameAdapter = srcMount.adapter === destMount.adapter;
146147
const createArgs = t => ({req, res, adapter: t.adapter, mount: t.mount});
147148

148-
await checkMountpointPermission(req, res, 'readfile', false)(srcMount);
149-
await checkMountpointPermission(req, res, 'writefile', true)(destMount);
149+
const srcStrict = srcMount.mount.attributes.strictGroups !== false;
150+
const destStrict = destMount.mount.attributes.strictGroups !== false;
151+
await checkMountpointPermission(req, res, 'readfile', false, srcStrict)(srcMount);
152+
await checkMountpointPermission(req, res, 'writefile', true, destStrict)(destMount);
150153

151154
if (sameAdapter) {
152155
const result = await srcMount

0 commit comments

Comments
 (0)
Please sign in to comment.