spaces.js 14.4 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
52
    if (req.query.search) {
      
53
54
55
      db.Membership.findAll({where:{
        user_id: req.user._id
      }}).then(memberships => {
mntmn's avatar
mntmn committed
56
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
84
            res.status(200).json(spaces);
          });
      });

    } else if (req.query.parent_space_id && req.query.parent_space_id != req.user.home_folder_id) {

85
86
      db.Space
        .findOne({where: {
mntmn's avatar
mntmn committed
87
          _id: req.query.parent_space_id
88
89
90
        }})
        //.populate('creator', userMapping)
        .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
104
                  }, include:['creator']})
                  .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
119
120
121
      db.Membership.findAll({ where: {
        user_id: req.user._id
      }}).then(memberships => {
        if (!memberships) memberships = [];
        
mntmn's avatar
mntmn committed
122
        var validMemberships = memberships.filter(function(m) {
123
          if (!m.space_id || (m.space_id == "undefined"))
mntmn's avatar
mntmn committed
124
125
126
127
            return false;
        });

        var spaceIds = validMemberships.map(function(m) {
128
          return m.space_id;
mntmn's avatar
mntmn committed
129
130
131
        });

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

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

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

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

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

221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
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
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
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'];

261
  db.Space.update(newAttr, {where: {
mntmn's avatar
mntmn committed
262
    "_id": space._id
263
264
  }}).then(space => {
    res.distributeUpdate("Space", space);
mntmn's avatar
mntmn committed
265
266
267
268
269
270
  });
});

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

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

var handleDuplicateSpaceRequest = function(req, res, parentSpace) {
  Space.duplicateSpace(req.space, req.user, 0, (err, newSpace) => {
    if (err) {
      console.error(err);
      res.status(400).json(err);
    } else {
      res.status(201).json(newSpace);
    }
  }, parentSpace);
}

router.post('/:id/duplicate', (req, res, next) => {
  if (req.query.parent_space_id) {
    Space.findOne({
      _id: req.query.parent_space_id
    }).populate('creator', userMapping).exec((err, parentSpace) => {
      if (!parentSpace) {
        res.status(404).json({
324
          "error": "parent space not found for duplicate"
mntmn's avatar
mntmn committed
325
326
        });
      } else {
327
        db.getUserRoleInSpace(parentSpace, req.user, (role) => {
mntmn's avatar
mntmn committed
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
          if (role == "admin" ||  role == "editor") {
            handleDuplicateSpaceRequest(req, res, parentSpace);
          } else {
            res.status(403).json({
              "error": "not authed for parent_space_id"
            });
          }
        });
      }
    });
  } else {
    handleDuplicateSpaceRequest(req, res);
  }
});

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

    if (req.spaceRole == "admin") {
      const attrs = req.body;
349
350
      space.destroy().then(function() {
        res.distributeDelete("Space", space);
mntmn's avatar
mntmn committed
351
352
353
      });
    } else {
      res.status(403).json({
354
        "error": "requires admin role"
mntmn's avatar
mntmn committed
355
356
357
358
359
360
361
362
363
364
365
366
      });
    }
  } else {
    res.sendStatus(403);
  }
});

router.post('/:id/artifacts-pdf', function(req, res, next) {
  if (req.spaceRole == "editor" || req.spaceRole == "admin") {

    var withZones = (req.query.zones) ? req.query.zones == "true" : false;
    var fileName = (req.query.filename || "upload.bin").replace(/[^a-zA-Z0-9\.]/g, '');
Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
367
    var localFilePath = os.tmpdir() + "/" + fileName;
mntmn's avatar
mntmn committed
368
369
370
371
372
373
    var writeStream = fs.createWriteStream(localFilePath);
    var stream = req.pipe(writeStream);

    req.on('end', function() {

      var rawName = fileName.slice(0, fileName.length - 4);
Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
374
375
376
      var outputFolder = os.tmpdir() + "/" + rawName;
      
      fs.mkdir(outputFolder, function(err) {
mntmn's avatar
mntmn committed
377
378
        var images = outputFolder + "/" + rawName + "-page-%03d.jpeg";
        
379
        // FIXME not portable
mntmn's avatar
mntmn committed
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
        exec.execFile("gs", ["-sDEVICE=jpeg", "-dDownScaleFactor=4", "-dDOINTERPOLATE", "-dNOPAUSE", "-dJPEGQ=80", "-dBATCH", "-sOutputFile=" + images, "-r250", "-f", localFilePath], {}, function(error, stdout, stderr) {
          if (error === null) {

            glob(outputFolder + "/*.jpeg", function(er, files) {
              var count = files.length;
              var delta = 10;

              var limitPerRow = Math.ceil(Math.sqrt(count));

              var startX = parseInt(req.query.x, delta);
              var startY = parseInt(req.query.y, delta);

              async.mapLimit(files, 20, function(localfilePath, cb) {

                var fileName = path.basename(localfilePath);
                var baseName = path.basename(localfilePath, ".jpeg");

                var number = parseInt(baseName.slice(baseName.length - 3, baseName.length), 10);

Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
399
                gm(localFilePath).size((err, size) => {
mntmn's avatar
mntmn committed
400
401
402
403
404
405
406
                  var w = 350;
                  var h = w;

                  var x = startX + (((number - 1) % limitPerRow) * w);
                  var y = startY + ((parseInt(((number - 1) / limitPerRow), 10) + 1) * w);

                  var userId;
Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
407
                  if (req.user) userId = req.user._id;
mntmn's avatar
mntmn committed
408

Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
409
410
                  var a = db.Artifact.create({
                    _id: uuidv4(),
mntmn's avatar
mntmn committed
411
412
413
414
                    mime: "image/jpg",
                    space_id: req.space._id,
                    user_id: userId,
                    editor_name: req.guest_name,
Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
                    w: w,
                    h: h,
                    x: x,
                    y: y,
                    z: (number + (count + 100))
                  }).then(a => {
                    payloadConverter.convert(a, fileName, localfilePath, (error, artifact) => {
                      if (error) res.status(400).json(error);
                      else {
                        if (withZones) {
                          var zone = {
                            _id: uuidv4(),
                            mime: "x-spacedeck/zone",
                            description: "Zone " + (number),
                            space_id: req.space._id,
                            user_id: userId,
                            editor_name: req.guest_name,
432
433
                            w: artifact.w + 20,
                            h: artifact.h + 40,
mntmn's avatar
mntmn committed
434
435
                            x: x - 10,
                            y: y - 30,
Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
436
                            z: number,
mntmn's avatar
mntmn committed
437
438
439
                            order: number,
                            valign: "middle",
                            align: "center"
Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
440
                          };
mntmn's avatar
mntmn committed
441

442
443
                          db.Artifact.create(zone).then((z) => {
                            redis.sendMessage("create", "Artifact", z.toJSON(), req.channelId);
Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
444
                            cb(null, [artifact, zone]);
445
                          });
mntmn's avatar
mntmn committed
446

Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
447
448
449
                        } else {
                          cb(null, [artifact]);
                        }
mntmn's avatar
mntmn committed
450
                      }
Lukas F. Hartmann's avatar
Lukas F. Hartmann committed
451
                    });
mntmn's avatar
mntmn committed
452
453
454
455
456
457
                  });

                });

              }, function(err, artifacts) {

458
                // FIXME not portable
mntmn's avatar
mntmn committed
459
460
461
462
463
464
465
                exec.execFile("rm", ["-r", outputFolder], function(err) {
                  res.status(201).json(_.flatten(artifacts));
                  
                  async.eachLimit(artifacts, 10, (artifact_or_artifacts, cb) => {

                    if (artifact_or_artifacts instanceof Array) {
                      _.each(artifact_or_artifacts, (a) => {
466
                        redis.sendMessage("create", "Artifact", JSON.stringify(a), req.channelId);
mntmn's avatar
mntmn committed
467
468
                      });
                    } else  {
469
                      redis.sendMessage("create", "Artifact", JSON.stringify(artifact_or_artifacts), req.channelId);
mntmn's avatar
mntmn committed
470
471
472
473
474
475
476
477
                    }
                    cb(null);
                  });
                });
              });
            });
          } else {
            console.error("error:", error);
478
            // FIXME not portable
mntmn's avatar
mntmn committed
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
            exec.execFile("rm", ["-r", outputFolder], function(err) {
              fs.unlink(localFilePath);
              res.status(400).json({});
            });
          }
        });
      });
    });
  } else {
    res.status(401).json({
      "error": "no access"
    });
  }
});

module.exports = router;