spaces.js 8.97 KB
Newer Older
mntmn's avatar
mntmn committed
1
2
"use strict";
var config = require('config');
Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
3
const os = require('os');
4
5
6
7
const db = require('../../models/db');
const Sequelize = require('sequelize');
const Op = Sequelize.Op;
const uuidv4 = require('uuid/v4');
mntmn's avatar
mntmn committed
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

var redis = require('../../helpers/redis');
var mailer = require('../../helpers/mailer');
var uploader = require('../../helpers/uploader');
var space_render = require('../../helpers/space-render');
var phantom = require('../../helpers/phantom');
var payloadConverter = require('../../helpers/artifact_converter');

var slug = require('slug');

var fs = require('fs');
var async = require('async');
var _ = require("underscore");
var request = require('request');
var url = require("url");
var path = require("path");
var crypto = require('crypto');
var glob = require('glob');
var gm = require('gm');
const exec = require('child_process');
var express = require('express');
var router = express.Router();

// JSON MAPPINGS
var userMapping = {
  _id: 1,
  nickname: 1,
  email: 1,
  avatar_thumb_uri: 1
};

var spaceMapping = {
  _id: 1,
  name: 1,
  thumbnail_url: 1
};

router.get('/', function(req, res, next) {
  if (!req.user) {
    res.status(403).json({
      error: "auth required"
    });
  } else {
51
    if (req.query.search) {
52
53
54
      db.Membership.findAll({where:{
        user_id: req.user._id
      }}).then(memberships => {
55
56
        // search for spaces

mntmn's avatar
mntmn committed
57
        var validMemberships = memberships.filter(function(m) {
58
          if (!m.space_id || (m.space_id == "undefined"))
mntmn's avatar
mntmn committed
59
60
            return false;
          else
61
            return true;
mntmn's avatar
mntmn committed
62
63
64
        });

        var spaceIds = validMemberships.map(function(m) {
65
          return m.space_id;
mntmn's avatar
mntmn committed
66
67
        });

68
69
70
71
72
73
        // TODO FIXME port
        var q = { where: {
          [Op.or]: [{"creator_id": req.user._id},
                   {"_id": {[Op.in]: spaceIds}},
                   {"parent_space_id": {[Op.in]: spaceIds}}],
          name: {[Op.like]: "%"+req.query.search+"%"}
74
        }, include: [db.CreatorSafeInclude(db)]};
75
76
77
78

        db.Space
          .findAll(q)
          .then(function(spaces) {
mntmn's avatar
mntmn committed
79
80
81
82
83
            res.status(200).json(spaces);
          });
      });

    } else if (req.query.parent_space_id && req.query.parent_space_id != req.user.home_folder_id) {
84
85
      // list spaces in a folder
      
86
87
      db.Space
        .findOne({where: {
mntmn's avatar
mntmn committed
88
          _id: req.query.parent_space_id
89
90
        }})
        .then(function(space) {
mntmn's avatar
mntmn committed
91
          if (space) {
92
            db.getUserRoleInSpace(space, req.user, function(role) {
mntmn's avatar
mntmn committed
93
              if (role == "none") {
94
                if (space.access_mode == "public") {
mntmn's avatar
mntmn committed
95
96
97
98
99
                  role = "viewer";
                }
              }

              if (role != "none") {
100
101
                db.Space
                  .findAll({where:{
mntmn's avatar
mntmn committed
102
                    parent_space_id: req.query.parent_space_id
103
                  }, include:[db.CreatorSafeInclude(db)]})
104
                  .then(function(spaces) {
mntmn's avatar
mntmn committed
105
106
107
108
109
110
111
112
113
114
115
116
                    res.status(200).json(spaces);
                  });
              } else {
                res.status(403).json({"error": "no authorized"});
              }
            });
          } else {
            res.status(404).json({"error": "space not found"});
          }
        });

    } else {
117
118
      // list home folder and spaces/folders that the user is a member of
      
119
120
121
122
      db.Membership.findAll({ where: {
        user_id: req.user._id
      }}).then(memberships => {
        if (!memberships) memberships = [];
123

mntmn's avatar
mntmn committed
124
        var validMemberships = memberships.filter(function(m) {
125
          if (!m.space_id || (m.space_id == "undefined"))
mntmn's avatar
mntmn committed
126
            return false;
127
          return true;
mntmn's avatar
mntmn committed
128
129
130
        });

        var spaceIds = validMemberships.map(function(m) {
131
          return m.space_id;
mntmn's avatar
mntmn committed
132
133
134
        });

        var q = {
135
136
          [Op.or]: [{
            "creator_id": req.user._id,
mntmn's avatar
mntmn committed
137
138
139
            "parent_space_id": req.user.home_folder_id
          }, {
            "_id": {
140
              [Op.in]: spaceIds
mntmn's avatar
mntmn committed
141
            },
142
143
            "creator_id": {
              [Op.ne]: req.user._id
mntmn's avatar
mntmn committed
144
145
146
147
            }
          }]
        };

148
        db.Space
149
          .findAll({where: q, include: [db.CreatorSafeInclude(db)]})
150
          .then(function(spaces) {
mntmn's avatar
mntmn committed
151
            var updatedSpaces = spaces.map(function(s) {
152
              var spaceObj = db.spaceToObject(s);
mntmn's avatar
mntmn committed
153
154
155
156
157
158
159
160
161
              return spaceObj;
            });
            res.status(200).json(spaces);
          });
      });
    }
  }
});

162
// create a space
mntmn's avatar
mntmn committed
163
164
165
166
167
router.post('/', function(req, res, next) {
  if (req.user) {
    var attrs = req.body;

    var createSpace = () => {
168
169
      attrs._id = uuidv4();
      attrs.creator_id = req.user._id;
mntmn's avatar
mntmn committed
170
      attrs.edit_hash = crypto.randomBytes(64).toString('hex').substring(0, 7);
171
      attrs.edit_slug = attrs.edit_slug || slug(attrs.name);
mntmn's avatar
mntmn committed
172
      attrs.access_mode = "private";
mntmn's avatar
mntmn committed
173
      
174
      db.Space.create(attrs).then(createdSpace => {
175
176
177
        res.status(201).json(createdSpace);
        
        // create initial admin membership
178
179
180
181
        var membership = {
          _id: uuidv4(),
          user_id: req.user._id,
          space_id: attrs._id,
182
183
          role: "admin",
          state: "active"
184
185
186
187
188
        };
        
        db.Membership.create(membership).then(() => {
          res.status(201).json(createdSpace);
        });
mntmn's avatar
mntmn committed
189
190
191
192
      });
    }

    if (attrs.parent_space_id) {
193
      db.Space.findOne({ where: {
mntmn's avatar
mntmn committed
194
        "_id": attrs.parent_space_id
195
      }}).then(parentSpace => {
mntmn's avatar
mntmn committed
196
        if (parentSpace) {
197
          db.getUserRoleInSpace(parentSpace, req.user, (role) => {
mntmn's avatar
mntmn committed
198
199
200
201
            if ((role == "editor") || (role == "admin")) {
              createSpace();
            } else {
              res.status(403).json({
202
                "error": "not editor in parent Space. role: "+role
mntmn's avatar
mntmn committed
203
204
205
206
207
208
209
210
211
212
              });
            }
          });
        } else {
          res.status(404).json({
            "error": "parent Space not found"
          });
        }
      });
    } else {
213
      attrs.parent_space_id = req.user.home_folder_id;
mntmn's avatar
mntmn committed
214
215
216
217
218
219
220
221
222
223
224
225
      createSpace();
    }

  } else {
    res.sendStatus(403);
  }
});

router.get('/:id', function(req, res, next) {
  res.status(200).json(req.space);
});

226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
router.get('/:id/path', (req, res) => {
  // build up a breadcrumb trail (path)
  var path = [];
  var buildPath = (space) => {
    if (space.parent_space_id) {
      db.Space.findOne({ where: {
        "_id": space.parent_space_id
      }}).then(parentSpace => {
        if (space._id == parentSpace._id) {
          console.error("error: circular parent reference for space " + space._id);
          res.send("error: circular reference");
        } else {
          path.push(parentSpace);
          buildPath(parentSpace);
        }
      });
    } else {
      // reached the top
      res.json(path.reverse());
    }
  }
  buildPath(req.space);
});

mntmn's avatar
mntmn committed
250
251
252
253
254
255
256
257
258
259
260
261
262
263
router.put('/:id', function(req, res) {
  var space = req.space;
  var newAttr = req.body;

  if (req['spaceRole'] != "editor" && req['spaceRole'] != "admin") {
    res.sendStatus(403);
    return;
  }

  newAttr.updated_at = new Date();
  newAttr.edit_slug = slug(newAttr['name']);

  delete newAttr['_id'];
  delete newAttr['creator'];
mntmn's avatar
mntmn committed
264
265
266
267
268
269
270
271
272
273
  delete newAttr['creator_id'];
  delete newAttr['space_type'];

  if (req['spaceRole'] != "admin") {
    delete newAttr['access_mode']
    delete newAttr['password']
    delete newAttr['edit_hash']
    delete newAttr['edit_slug']
    delete newAttr['editors_locking']
  }
mntmn's avatar
mntmn committed
274

275
  db.Space.update(newAttr, {where: {
mntmn's avatar
mntmn committed
276
    "_id": space._id
277
278
  }}).then(space => {
    res.distributeUpdate("Space", space);
mntmn's avatar
mntmn committed
279
280
281
282
283
284
  });
});

router.post('/:id/background', function(req, res, next) {
  var space = req.space;
  var newDate = new Date();
285
  var fileName = (req.query.filename || "upload.jpg").replace(/[^a-zA-Z0-9\.]/g, '');
mntmn's avatar
mntmn committed
286
287
288
289
290
291
292
293
  var localFilePath = "/tmp/" + fileName;
  var writeStream = fs.createWriteStream(localFilePath);
  var stream = req.pipe(writeStream);

  req.on('end', function() {
    uploader.uploadFile("s" + req.space._id + "/bg_" + newDate.getTime() + "_" + fileName, "image/jpeg", localFilePath, function(err, backgroundUrl) {
      if (err) res.status(400).json(err);
      else {
294
295
        if (space.background_uri) {
          var oldPath = url.parse(req.space.background_uri).pathname;
mntmn's avatar
mntmn committed
296
          uploader.removeFile(oldPath, function(err) {
297
            console.error("removed old bg error:", err);
mntmn's avatar
mntmn committed
298
299
300
          });
        }

301
302
        db.Space.update({
          background_uri: backgroundUrl
mntmn's avatar
mntmn committed
303
        }, {
304
305
306
307
308
309
310
311
312
313
          where: { "_id": space._id }
        }, function(rows) {
          fs.unlink(localFilePath, function(err) {
            if (err) {
              console.error(err);
              res.status(400).json(err);
            } else {
              res.status(200).json(space);
            }
          });
mntmn's avatar
mntmn committed
314
315
316
317
318
319
320
321
322
323
324
325
        });
      }
    });
  });
});

router.delete('/:id', function(req, res, next) {
  if (req.user) {
    const space = req.space;

    if (req.spaceRole == "admin") {
      const attrs = req.body;
326
327
      space.destroy().then(function() {
        res.distributeDelete("Space", space);
mntmn's avatar
mntmn committed
328
329
330
      });
    } else {
      res.status(403).json({
331
        "error": "requires admin role"
mntmn's avatar
mntmn committed
332
333
334
335
336
337
338
339
      });
    }
  } else {
    res.sendStatus(403);
  }
});

module.exports = router;