spaces.js 8.66 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
74
75
76
77
78
        // 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+"%"}
        }, include: ['creator']};

        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
91
        }})
        //.populate('creator', userMapping)
        .then(function(space) {
mntmn's avatar
mntmn committed
92
          if (space) {
93
            db.getUserRoleInSpace(space, req.user, function(role) {
mntmn's avatar
mntmn committed
94
              if (role == "none") {
95
                if (space.access_mode == "public") {
mntmn's avatar
mntmn committed
96
97
98
99
100
                  role = "viewer";
                }
              }

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

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

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

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

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

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

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

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

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

266
  db.Space.update(newAttr, {where: {
mntmn's avatar
mntmn committed
267
    "_id": space._id
268
269
  }}).then(space => {
    res.distributeUpdate("Space", space);
mntmn's avatar
mntmn committed
270
271
272
273
274
275
  });
});

router.post('/:id/background', function(req, res, next) {
  var space = req.space;
  var newDate = new Date();
276
  var fileName = (req.query.filename || "upload.jpg").replace(/[^a-zA-Z0-9\.]/g, '');
mntmn's avatar
mntmn committed
277
278
279
280
281
282
283
284
  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 {
285
286
        if (space.background_uri) {
          var oldPath = url.parse(req.space.background_uri).pathname;
mntmn's avatar
mntmn committed
287
          uploader.removeFile(oldPath, function(err) {
288
            console.error("removed old bg error:", err);
mntmn's avatar
mntmn committed
289
290
291
          });
        }

292
293
        db.Space.update({
          background_uri: backgroundUrl
mntmn's avatar
mntmn committed
294
        }, {
295
296
297
298
299
300
301
302
303
304
          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
305
306
307
308
309
310
311
312
313
314
315
316
        });
      }
    });
  });
});

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

    if (req.spaceRole == "admin") {
      const attrs = req.body;
317
318
      space.destroy().then(function() {
        res.distributeDelete("Space", space);
mntmn's avatar
mntmn committed
319
320
321
      });
    } else {
      res.status(403).json({
322
        "error": "requires admin role"
mntmn's avatar
mntmn committed
323
324
325
326
327
328
329
330
      });
    }
  } else {
    res.sendStatus(403);
  }
});

module.exports = router;