Use JavaScript Standard Style
Introduce JavaScript Standard Style as project style rule, and fixed all fail on backend code.
This commit is contained in:
		
							parent
							
								
									8f1c97f4a4
								
							
						
					
					
						commit
						4889e9732d
					
				
							
								
								
									
										360
									
								
								lib/auth.js
									
									
									
									
									
								
							
							
						
						
									
										360
									
								
								lib/auth.js
									
									
									
									
									
								
							@ -1,190 +1,192 @@
 | 
			
		||||
//auth
 | 
			
		||||
//external modules
 | 
			
		||||
var passport = require('passport');
 | 
			
		||||
var FacebookStrategy = require('passport-facebook').Strategy;
 | 
			
		||||
var TwitterStrategy = require('passport-twitter').Strategy;
 | 
			
		||||
var GithubStrategy = require('passport-github').Strategy;
 | 
			
		||||
var GitlabStrategy = require('passport-gitlab2').Strategy;
 | 
			
		||||
var DropboxStrategy = require('passport-dropbox-oauth2').Strategy;
 | 
			
		||||
var GoogleStrategy = require('passport-google-oauth20').Strategy;
 | 
			
		||||
var LdapStrategy = require('passport-ldapauth');
 | 
			
		||||
var LocalStrategy = require('passport-local').Strategy;
 | 
			
		||||
var validator = require('validator');
 | 
			
		||||
// auth
 | 
			
		||||
// external modules
 | 
			
		||||
var passport = require('passport')
 | 
			
		||||
var FacebookStrategy = require('passport-facebook').Strategy
 | 
			
		||||
var TwitterStrategy = require('passport-twitter').Strategy
 | 
			
		||||
var GithubStrategy = require('passport-github').Strategy
 | 
			
		||||
var GitlabStrategy = require('passport-gitlab2').Strategy
 | 
			
		||||
var DropboxStrategy = require('passport-dropbox-oauth2').Strategy
 | 
			
		||||
var GoogleStrategy = require('passport-google-oauth20').Strategy
 | 
			
		||||
var LdapStrategy = require('passport-ldapauth')
 | 
			
		||||
var LocalStrategy = require('passport-local').Strategy
 | 
			
		||||
var validator = require('validator')
 | 
			
		||||
 | 
			
		||||
//core
 | 
			
		||||
var config = require('./config.js');
 | 
			
		||||
var logger = require("./logger.js");
 | 
			
		||||
var models = require("./models");
 | 
			
		||||
// core
 | 
			
		||||
var config = require('./config.js')
 | 
			
		||||
var logger = require('./logger.js')
 | 
			
		||||
var models = require('./models')
 | 
			
		||||
 | 
			
		||||
function callback(accessToken, refreshToken, profile, done) {
 | 
			
		||||
    //logger.info(profile.displayName || profile.username);
 | 
			
		||||
    var stringifiedProfile = JSON.stringify(profile);
 | 
			
		||||
    models.User.findOrCreate({
 | 
			
		||||
function callback (accessToken, refreshToken, profile, done) {
 | 
			
		||||
  // logger.info(profile.displayName || profile.username);
 | 
			
		||||
  var stringifiedProfile = JSON.stringify(profile)
 | 
			
		||||
  models.User.findOrCreate({
 | 
			
		||||
    where: {
 | 
			
		||||
      profileid: profile.id.toString()
 | 
			
		||||
    },
 | 
			
		||||
    defaults: {
 | 
			
		||||
      profile: stringifiedProfile,
 | 
			
		||||
      accessToken: accessToken,
 | 
			
		||||
      refreshToken: refreshToken
 | 
			
		||||
    }
 | 
			
		||||
  }).spread(function (user, created) {
 | 
			
		||||
    if (user) {
 | 
			
		||||
      var needSave = false
 | 
			
		||||
      if (user.profile !== stringifiedProfile) {
 | 
			
		||||
        user.profile = stringifiedProfile
 | 
			
		||||
        needSave = true
 | 
			
		||||
      }
 | 
			
		||||
      if (user.accessToken !== accessToken) {
 | 
			
		||||
        user.accessToken = accessToken
 | 
			
		||||
        needSave = true
 | 
			
		||||
      }
 | 
			
		||||
      if (user.refreshToken !== refreshToken) {
 | 
			
		||||
        user.refreshToken = refreshToken
 | 
			
		||||
        needSave = true
 | 
			
		||||
      }
 | 
			
		||||
      if (needSave) {
 | 
			
		||||
        user.save().then(function () {
 | 
			
		||||
          if (config.debug) { logger.info('user login: ' + user.id) }
 | 
			
		||||
          return done(null, user)
 | 
			
		||||
        })
 | 
			
		||||
      } else {
 | 
			
		||||
        if (config.debug) { logger.info('user login: ' + user.id) }
 | 
			
		||||
        return done(null, user)
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }).catch(function (err) {
 | 
			
		||||
    logger.error('auth callback failed: ' + err)
 | 
			
		||||
    return done(err, null)
 | 
			
		||||
  })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function registerAuthMethod () {
 | 
			
		||||
// facebook
 | 
			
		||||
  if (config.facebook) {
 | 
			
		||||
    passport.use(new FacebookStrategy({
 | 
			
		||||
      clientID: config.facebook.clientID,
 | 
			
		||||
      clientSecret: config.facebook.clientSecret,
 | 
			
		||||
      callbackURL: config.serverurl + '/auth/facebook/callback'
 | 
			
		||||
    }, callback))
 | 
			
		||||
  }
 | 
			
		||||
// twitter
 | 
			
		||||
  if (config.twitter) {
 | 
			
		||||
    passport.use(new TwitterStrategy({
 | 
			
		||||
      consumerKey: config.twitter.consumerKey,
 | 
			
		||||
      consumerSecret: config.twitter.consumerSecret,
 | 
			
		||||
      callbackURL: config.serverurl + '/auth/twitter/callback'
 | 
			
		||||
    }, callback))
 | 
			
		||||
  }
 | 
			
		||||
// github
 | 
			
		||||
  if (config.github) {
 | 
			
		||||
    passport.use(new GithubStrategy({
 | 
			
		||||
      clientID: config.github.clientID,
 | 
			
		||||
      clientSecret: config.github.clientSecret,
 | 
			
		||||
      callbackURL: config.serverurl + '/auth/github/callback'
 | 
			
		||||
    }, callback))
 | 
			
		||||
  }
 | 
			
		||||
// gitlab
 | 
			
		||||
  if (config.gitlab) {
 | 
			
		||||
    passport.use(new GitlabStrategy({
 | 
			
		||||
      baseURL: config.gitlab.baseURL,
 | 
			
		||||
      clientID: config.gitlab.clientID,
 | 
			
		||||
      clientSecret: config.gitlab.clientSecret,
 | 
			
		||||
      callbackURL: config.serverurl + '/auth/gitlab/callback'
 | 
			
		||||
    }, callback))
 | 
			
		||||
  }
 | 
			
		||||
// dropbox
 | 
			
		||||
  if (config.dropbox) {
 | 
			
		||||
    passport.use(new DropboxStrategy({
 | 
			
		||||
      apiVersion: '2',
 | 
			
		||||
      clientID: config.dropbox.clientID,
 | 
			
		||||
      clientSecret: config.dropbox.clientSecret,
 | 
			
		||||
      callbackURL: config.serverurl + '/auth/dropbox/callback'
 | 
			
		||||
    }, callback))
 | 
			
		||||
  }
 | 
			
		||||
// google
 | 
			
		||||
  if (config.google) {
 | 
			
		||||
    passport.use(new GoogleStrategy({
 | 
			
		||||
      clientID: config.google.clientID,
 | 
			
		||||
      clientSecret: config.google.clientSecret,
 | 
			
		||||
      callbackURL: config.serverurl + '/auth/google/callback'
 | 
			
		||||
    }, callback))
 | 
			
		||||
  }
 | 
			
		||||
// ldap
 | 
			
		||||
  if (config.ldap) {
 | 
			
		||||
    passport.use(new LdapStrategy({
 | 
			
		||||
      server: {
 | 
			
		||||
        url: config.ldap.url || null,
 | 
			
		||||
        bindDn: config.ldap.bindDn || null,
 | 
			
		||||
        bindCredentials: config.ldap.bindCredentials || null,
 | 
			
		||||
        searchBase: config.ldap.searchBase || null,
 | 
			
		||||
        searchFilter: config.ldap.searchFilter || null,
 | 
			
		||||
        searchAttributes: config.ldap.searchAttributes || null,
 | 
			
		||||
        tlsOptions: config.ldap.tlsOptions || null
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    function (user, done) {
 | 
			
		||||
      var profile = {
 | 
			
		||||
        id: 'LDAP-' + user.uidNumber,
 | 
			
		||||
        username: user.uid,
 | 
			
		||||
        displayName: user.displayName,
 | 
			
		||||
        emails: user.mail ? [user.mail] : [],
 | 
			
		||||
        avatarUrl: null,
 | 
			
		||||
        profileUrl: null,
 | 
			
		||||
        provider: 'ldap'
 | 
			
		||||
      }
 | 
			
		||||
      var stringifiedProfile = JSON.stringify(profile)
 | 
			
		||||
      models.User.findOrCreate({
 | 
			
		||||
        where: {
 | 
			
		||||
            profileid: profile.id.toString()
 | 
			
		||||
          profileid: profile.id.toString()
 | 
			
		||||
        },
 | 
			
		||||
        defaults: {
 | 
			
		||||
            profile: stringifiedProfile,
 | 
			
		||||
            accessToken: accessToken,
 | 
			
		||||
            refreshToken: refreshToken
 | 
			
		||||
          profile: stringifiedProfile
 | 
			
		||||
        }
 | 
			
		||||
    }).spread(function (user, created) {
 | 
			
		||||
      }).spread(function (user, created) {
 | 
			
		||||
        if (user) {
 | 
			
		||||
            var needSave = false;
 | 
			
		||||
            if (user.profile != stringifiedProfile) {
 | 
			
		||||
                user.profile = stringifiedProfile;
 | 
			
		||||
                needSave = true;
 | 
			
		||||
            }
 | 
			
		||||
            if (user.accessToken != accessToken) {
 | 
			
		||||
                user.accessToken = accessToken;
 | 
			
		||||
                needSave = true;
 | 
			
		||||
            }
 | 
			
		||||
            if (user.refreshToken != refreshToken) {
 | 
			
		||||
                user.refreshToken = refreshToken;
 | 
			
		||||
                needSave = true;
 | 
			
		||||
            }
 | 
			
		||||
            if (needSave) {
 | 
			
		||||
                user.save().then(function () {
 | 
			
		||||
                    if (config.debug)
 | 
			
		||||
                        logger.info('user login: ' + user.id);
 | 
			
		||||
                    return done(null, user);
 | 
			
		||||
                });
 | 
			
		||||
            } else {
 | 
			
		||||
                if (config.debug)
 | 
			
		||||
                    logger.info('user login: ' + user.id);
 | 
			
		||||
                return done(null, user);
 | 
			
		||||
            }
 | 
			
		||||
          var needSave = false
 | 
			
		||||
          if (user.profile !== stringifiedProfile) {
 | 
			
		||||
            user.profile = stringifiedProfile
 | 
			
		||||
            needSave = true
 | 
			
		||||
          }
 | 
			
		||||
          if (needSave) {
 | 
			
		||||
            user.save().then(function () {
 | 
			
		||||
              if (config.debug) { logger.info('user login: ' + user.id) }
 | 
			
		||||
              return done(null, user)
 | 
			
		||||
            })
 | 
			
		||||
          } else {
 | 
			
		||||
            if (config.debug) { logger.info('user login: ' + user.id) }
 | 
			
		||||
            return done(null, user)
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
    }).catch(function (err) {
 | 
			
		||||
        logger.error('auth callback failed: ' + err);
 | 
			
		||||
        return done(err, null);
 | 
			
		||||
    });
 | 
			
		||||
      }).catch(function (err) {
 | 
			
		||||
        logger.error('ldap auth failed: ' + err)
 | 
			
		||||
        return done(err, null)
 | 
			
		||||
      })
 | 
			
		||||
    }))
 | 
			
		||||
  }
 | 
			
		||||
// email
 | 
			
		||||
  if (config.email) {
 | 
			
		||||
    passport.use(new LocalStrategy({
 | 
			
		||||
      usernameField: 'email'
 | 
			
		||||
    },
 | 
			
		||||
    function (email, password, done) {
 | 
			
		||||
      if (!validator.isEmail(email)) return done(null, false)
 | 
			
		||||
      models.User.findOne({
 | 
			
		||||
        where: {
 | 
			
		||||
          email: email
 | 
			
		||||
        }
 | 
			
		||||
      }).then(function (user) {
 | 
			
		||||
        if (!user) return done(null, false)
 | 
			
		||||
        if (!user.verifyPassword(password)) return done(null, false)
 | 
			
		||||
        return done(null, user)
 | 
			
		||||
      }).catch(function (err) {
 | 
			
		||||
        logger.error(err)
 | 
			
		||||
        return done(err)
 | 
			
		||||
      })
 | 
			
		||||
    }))
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
//facebook
 | 
			
		||||
if (config.facebook) {
 | 
			
		||||
    module.exports = passport.use(new FacebookStrategy({
 | 
			
		||||
        clientID: config.facebook.clientID,
 | 
			
		||||
        clientSecret: config.facebook.clientSecret,
 | 
			
		||||
        callbackURL: config.serverurl + '/auth/facebook/callback'
 | 
			
		||||
    }, callback));
 | 
			
		||||
}
 | 
			
		||||
//twitter
 | 
			
		||||
if (config.twitter) {
 | 
			
		||||
    passport.use(new TwitterStrategy({
 | 
			
		||||
        consumerKey: config.twitter.consumerKey,
 | 
			
		||||
        consumerSecret: config.twitter.consumerSecret,
 | 
			
		||||
        callbackURL: config.serverurl + '/auth/twitter/callback'
 | 
			
		||||
    }, callback));
 | 
			
		||||
}
 | 
			
		||||
//github
 | 
			
		||||
if (config.github) {
 | 
			
		||||
    passport.use(new GithubStrategy({
 | 
			
		||||
        clientID: config.github.clientID,
 | 
			
		||||
        clientSecret: config.github.clientSecret,
 | 
			
		||||
        callbackURL: config.serverurl + '/auth/github/callback'
 | 
			
		||||
    }, callback));
 | 
			
		||||
}
 | 
			
		||||
//gitlab
 | 
			
		||||
if (config.gitlab) {
 | 
			
		||||
    passport.use(new GitlabStrategy({
 | 
			
		||||
        baseURL: config.gitlab.baseURL,
 | 
			
		||||
        clientID: config.gitlab.clientID,
 | 
			
		||||
        clientSecret: config.gitlab.clientSecret,
 | 
			
		||||
        callbackURL: config.serverurl + '/auth/gitlab/callback'
 | 
			
		||||
    }, callback));
 | 
			
		||||
}
 | 
			
		||||
//dropbox
 | 
			
		||||
if (config.dropbox) {
 | 
			
		||||
    passport.use(new DropboxStrategy({
 | 
			
		||||
        apiVersion: '2',
 | 
			
		||||
        clientID: config.dropbox.clientID,
 | 
			
		||||
        clientSecret: config.dropbox.clientSecret,
 | 
			
		||||
        callbackURL: config.serverurl + '/auth/dropbox/callback'
 | 
			
		||||
    }, callback));
 | 
			
		||||
}
 | 
			
		||||
//google
 | 
			
		||||
if (config.google) {
 | 
			
		||||
    passport.use(new GoogleStrategy({
 | 
			
		||||
        clientID: config.google.clientID,
 | 
			
		||||
        clientSecret: config.google.clientSecret,
 | 
			
		||||
        callbackURL: config.serverurl + '/auth/google/callback'
 | 
			
		||||
    }, callback));
 | 
			
		||||
}
 | 
			
		||||
// ldap
 | 
			
		||||
if (config.ldap) {
 | 
			
		||||
    passport.use(new LdapStrategy({
 | 
			
		||||
        server: {
 | 
			
		||||
            url: config.ldap.url || null,
 | 
			
		||||
            bindDn: config.ldap.bindDn || null,
 | 
			
		||||
            bindCredentials: config.ldap.bindCredentials || null,
 | 
			
		||||
            searchBase: config.ldap.searchBase || null,
 | 
			
		||||
            searchFilter: config.ldap.searchFilter || null,
 | 
			
		||||
            searchAttributes: config.ldap.searchAttributes || null,
 | 
			
		||||
            tlsOptions: config.ldap.tlsOptions || null
 | 
			
		||||
        },
 | 
			
		||||
    },
 | 
			
		||||
    function(user, done) {
 | 
			
		||||
        var profile = {
 | 
			
		||||
            id: 'LDAP-' + user.uidNumber,
 | 
			
		||||
            username: user.uid,
 | 
			
		||||
            displayName: user.displayName,
 | 
			
		||||
            emails: user.mail ? [user.mail] : [],
 | 
			
		||||
            avatarUrl: null,
 | 
			
		||||
            profileUrl: null,
 | 
			
		||||
            provider: 'ldap',
 | 
			
		||||
        }
 | 
			
		||||
        var stringifiedProfile = JSON.stringify(profile);
 | 
			
		||||
        models.User.findOrCreate({
 | 
			
		||||
            where: {
 | 
			
		||||
                profileid: profile.id.toString()
 | 
			
		||||
            },
 | 
			
		||||
            defaults: {
 | 
			
		||||
                profile: stringifiedProfile,
 | 
			
		||||
            }
 | 
			
		||||
        }).spread(function (user, created) {
 | 
			
		||||
            if (user) {
 | 
			
		||||
                var needSave = false;
 | 
			
		||||
                if (user.profile != stringifiedProfile) {
 | 
			
		||||
                    user.profile = stringifiedProfile;
 | 
			
		||||
                    needSave = true;
 | 
			
		||||
                }
 | 
			
		||||
                if (needSave) {
 | 
			
		||||
                    user.save().then(function () {
 | 
			
		||||
                        if (config.debug)
 | 
			
		||||
                            logger.info('user login: ' + user.id);
 | 
			
		||||
                        return done(null, user);
 | 
			
		||||
                    });
 | 
			
		||||
                } else {
 | 
			
		||||
                    if (config.debug)
 | 
			
		||||
                        logger.info('user login: ' + user.id);
 | 
			
		||||
                    return done(null, user);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }).catch(function (err) {
 | 
			
		||||
            logger.error('ldap auth failed: ' + err);
 | 
			
		||||
            return done(err, null);
 | 
			
		||||
        });
 | 
			
		||||
    }));
 | 
			
		||||
}
 | 
			
		||||
// email
 | 
			
		||||
if (config.email) {
 | 
			
		||||
    passport.use(new LocalStrategy({
 | 
			
		||||
        usernameField: 'email'
 | 
			
		||||
    },
 | 
			
		||||
    function(email, password, done) {
 | 
			
		||||
        if (!validator.isEmail(email)) return done(null, false);
 | 
			
		||||
        models.User.findOne({
 | 
			
		||||
            where: {
 | 
			
		||||
                email: email
 | 
			
		||||
            }
 | 
			
		||||
        }).then(function (user) {
 | 
			
		||||
            if (!user) return done(null, false);
 | 
			
		||||
            if (!user.verifyPassword(password)) return done(null, false);
 | 
			
		||||
            return done(null, user);
 | 
			
		||||
        }).catch(function (err) {
 | 
			
		||||
            logger.error(err);
 | 
			
		||||
            return done(err);
 | 
			
		||||
        });
 | 
			
		||||
    }));
 | 
			
		||||
module.exports = {
 | 
			
		||||
  registerAuthMethod: registerAuthMethod
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										330
									
								
								lib/config.js
									
									
									
									
									
								
							
							
						
						
									
										330
									
								
								lib/config.js
									
									
									
									
									
								
							@ -1,118 +1,117 @@
 | 
			
		||||
// external modules
 | 
			
		||||
var fs = require('fs');
 | 
			
		||||
var path = require('path');
 | 
			
		||||
var fs = require('fs');
 | 
			
		||||
var fs = require('fs')
 | 
			
		||||
var path = require('path')
 | 
			
		||||
 | 
			
		||||
// configs
 | 
			
		||||
var env = process.env.NODE_ENV || 'development';
 | 
			
		||||
var config = require(path.join(__dirname, '..', 'config.json'))[env];
 | 
			
		||||
var debug = process.env.DEBUG ? (process.env.DEBUG === 'true') : ((typeof config.debug === 'boolean') ? config.debug : (env === 'development'));
 | 
			
		||||
var env = process.env.NODE_ENV || 'development'
 | 
			
		||||
var config = require(path.join(__dirname, '..', 'config.json'))[env]
 | 
			
		||||
var debug = process.env.DEBUG ? (process.env.DEBUG === 'true') : ((typeof config.debug === 'boolean') ? config.debug : (env === 'development'))
 | 
			
		||||
 | 
			
		||||
// Create function that reads docker secrets but fails fast in case of a non docker environment
 | 
			
		||||
var handleDockerSecret = fs.existsSync('/run/secrets/') ? function(secret) {
 | 
			
		||||
    return fs.existsSync('/run/secrets/' + secret) ? fs.readFileSync('/run/secrets/' + secret) : null;
 | 
			
		||||
} : function() {
 | 
			
		||||
    return null
 | 
			
		||||
};
 | 
			
		||||
var handleDockerSecret = fs.existsSync('/run/secrets/') ? function (secret) {
 | 
			
		||||
  return fs.existsSync('/run/secrets/' + secret) ? fs.readFileSync('/run/secrets/' + secret) : null
 | 
			
		||||
} : function () {
 | 
			
		||||
  return null
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// url
 | 
			
		||||
var domain = process.env.DOMAIN || process.env.HMD_DOMAIN || config.domain || '';
 | 
			
		||||
var urlpath = process.env.URL_PATH || process.env.HMD_URL_PATH || config.urlpath || '';
 | 
			
		||||
var port = process.env.PORT || process.env.HMD_PORT || config.port || 3000;
 | 
			
		||||
var alloworigin = process.env.HMD_ALLOW_ORIGIN ? process.env.HMD_ALLOW_ORIGIN.split(',') : (config.alloworigin || ['localhost']);
 | 
			
		||||
var domain = process.env.DOMAIN || process.env.HMD_DOMAIN || config.domain || ''
 | 
			
		||||
var urlpath = process.env.URL_PATH || process.env.HMD_URL_PATH || config.urlpath || ''
 | 
			
		||||
var port = process.env.PORT || process.env.HMD_PORT || config.port || 3000
 | 
			
		||||
var alloworigin = process.env.HMD_ALLOW_ORIGIN ? process.env.HMD_ALLOW_ORIGIN.split(',') : (config.alloworigin || ['localhost'])
 | 
			
		||||
 | 
			
		||||
var usessl = !!config.usessl;
 | 
			
		||||
var usessl = !!config.usessl
 | 
			
		||||
var protocolusessl = (usessl === true && typeof process.env.HMD_PROTOCOL_USESSL === 'undefined' && typeof config.protocolusessl === 'undefined')
 | 
			
		||||
     ? true : (process.env.HMD_PROTOCOL_USESSL ? (process.env.HMD_PROTOCOL_USESSL === 'true') : !!config.protocolusessl);
 | 
			
		||||
var urladdport = process.env.HMD_URL_ADDPORT ? (process.env.HMD_URL_ADDPORT === 'true') : !!config.urladdport;
 | 
			
		||||
     ? true : (process.env.HMD_PROTOCOL_USESSL ? (process.env.HMD_PROTOCOL_USESSL === 'true') : !!config.protocolusessl)
 | 
			
		||||
var urladdport = process.env.HMD_URL_ADDPORT ? (process.env.HMD_URL_ADDPORT === 'true') : !!config.urladdport
 | 
			
		||||
 | 
			
		||||
var usecdn = process.env.HMD_USECDN ? (process.env.HMD_USECDN === 'true') : ((typeof config.usecdn === 'boolean') ? config.usecdn : true);
 | 
			
		||||
var usecdn = process.env.HMD_USECDN ? (process.env.HMD_USECDN === 'true') : ((typeof config.usecdn === 'boolean') ? config.usecdn : true)
 | 
			
		||||
 | 
			
		||||
var allowanonymous = process.env.HMD_ALLOW_ANONYMOUS ? (process.env.HMD_ALLOW_ANONYMOUS === 'true') : ((typeof config.allowanonymous === 'boolean') ? config.allowanonymous : true);
 | 
			
		||||
var allowanonymous = process.env.HMD_ALLOW_ANONYMOUS ? (process.env.HMD_ALLOW_ANONYMOUS === 'true') : ((typeof config.allowanonymous === 'boolean') ? config.allowanonymous : true)
 | 
			
		||||
 | 
			
		||||
var allowfreeurl = process.env.HMD_ALLOW_FREEURL ? (process.env.HMD_ALLOW_FREEURL === 'true') : !!config.allowfreeurl;
 | 
			
		||||
var allowfreeurl = process.env.HMD_ALLOW_FREEURL ? (process.env.HMD_ALLOW_FREEURL === 'true') : !!config.allowfreeurl
 | 
			
		||||
 | 
			
		||||
var permissions = ['editable', 'limited', 'locked', 'protected', 'private'];
 | 
			
		||||
var permissions = ['editable', 'limited', 'locked', 'protected', 'private']
 | 
			
		||||
if (allowanonymous) {
 | 
			
		||||
    permissions.unshift('freely');
 | 
			
		||||
  permissions.unshift('freely')
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
var defaultpermission = process.env.HMD_DEFAULT_PERMISSION || config.defaultpermission;
 | 
			
		||||
defaultpermission = permissions.indexOf(defaultpermission) != -1 ? defaultpermission : 'editable';
 | 
			
		||||
var defaultpermission = process.env.HMD_DEFAULT_PERMISSION || config.defaultpermission
 | 
			
		||||
defaultpermission = permissions.indexOf(defaultpermission) !== -1 ? defaultpermission : 'editable'
 | 
			
		||||
 | 
			
		||||
// db
 | 
			
		||||
var dburl = process.env.HMD_DB_URL || process.env.DATABASE_URL || config.dburl;
 | 
			
		||||
var db = config.db || {};
 | 
			
		||||
var dburl = process.env.HMD_DB_URL || process.env.DATABASE_URL || config.dburl
 | 
			
		||||
var db = config.db || {}
 | 
			
		||||
 | 
			
		||||
// ssl path
 | 
			
		||||
var sslkeypath = (fs.existsSync('/run/secrets/key.pem') ? '/run/secrets/key.pem' : null) || config.sslkeypath || '';
 | 
			
		||||
var sslcertpath = (fs.existsSync('/run/secrets/cert.pem') ? '/run/secrets/cert.pem' : null) || config.sslcertpath || '';
 | 
			
		||||
var sslcapath = (fs.existsSync('/run/secrets/ca.pem') ? '/run/secrets/ca.pem' : null) || config.sslcapath || '';
 | 
			
		||||
var dhparampath = (fs.existsSync('/run/secrets/dhparam.pem') ? '/run/secrets/dhparam.pem' : null) || config.dhparampath || '';
 | 
			
		||||
var sslkeypath = (fs.existsSync('/run/secrets/key.pem') ? '/run/secrets/key.pem' : null) || config.sslkeypath || ''
 | 
			
		||||
var sslcertpath = (fs.existsSync('/run/secrets/cert.pem') ? '/run/secrets/cert.pem' : null) || config.sslcertpath || ''
 | 
			
		||||
var sslcapath = (fs.existsSync('/run/secrets/ca.pem') ? '/run/secrets/ca.pem' : null) || config.sslcapath || ''
 | 
			
		||||
var dhparampath = (fs.existsSync('/run/secrets/dhparam.pem') ? '/run/secrets/dhparam.pem' : null) || config.dhparampath || ''
 | 
			
		||||
 | 
			
		||||
// other path
 | 
			
		||||
var tmppath = config.tmppath || './tmp';
 | 
			
		||||
var defaultnotepath = config.defaultnotepath || './public/default.md';
 | 
			
		||||
var docspath = config.docspath || './public/docs';
 | 
			
		||||
var indexpath = config.indexpath || './public/views/index.ejs';
 | 
			
		||||
var hackmdpath = config.hackmdpath || './public/views/hackmd.ejs';
 | 
			
		||||
var errorpath = config.errorpath || './public/views/error.ejs';
 | 
			
		||||
var prettypath = config.prettypath || './public/views/pretty.ejs';
 | 
			
		||||
var slidepath = config.slidepath || './public/views/slide.ejs';
 | 
			
		||||
var tmppath = config.tmppath || './tmp'
 | 
			
		||||
var defaultnotepath = config.defaultnotepath || './public/default.md'
 | 
			
		||||
var docspath = config.docspath || './public/docs'
 | 
			
		||||
var indexpath = config.indexpath || './public/views/index.ejs'
 | 
			
		||||
var hackmdpath = config.hackmdpath || './public/views/hackmd.ejs'
 | 
			
		||||
var errorpath = config.errorpath || './public/views/error.ejs'
 | 
			
		||||
var prettypath = config.prettypath || './public/views/pretty.ejs'
 | 
			
		||||
var slidepath = config.slidepath || './public/views/slide.ejs'
 | 
			
		||||
 | 
			
		||||
// session
 | 
			
		||||
var sessionname = config.sessionname || 'connect.sid';
 | 
			
		||||
var sessionsecret = handleDockerSecret('sessionsecret') || config.sessionsecret || 'secret';
 | 
			
		||||
var sessionlife = config.sessionlife || 14 * 24 * 60 * 60 * 1000; //14 days
 | 
			
		||||
var sessionname = config.sessionname || 'connect.sid'
 | 
			
		||||
var sessionsecret = handleDockerSecret('sessionsecret') || config.sessionsecret || 'secret'
 | 
			
		||||
var sessionlife = config.sessionlife || 14 * 24 * 60 * 60 * 1000 // 14 days
 | 
			
		||||
 | 
			
		||||
// static files
 | 
			
		||||
var staticcachetime = config.staticcachetime || 1 * 24 * 60 * 60 * 1000; // 1 day
 | 
			
		||||
var staticcachetime = config.staticcachetime || 1 * 24 * 60 * 60 * 1000 // 1 day
 | 
			
		||||
 | 
			
		||||
// socket.io
 | 
			
		||||
var heartbeatinterval = config.heartbeatinterval || 5000;
 | 
			
		||||
var heartbeattimeout = config.heartbeattimeout || 10000;
 | 
			
		||||
var heartbeatinterval = config.heartbeatinterval || 5000
 | 
			
		||||
var heartbeattimeout = config.heartbeattimeout || 10000
 | 
			
		||||
 | 
			
		||||
// document
 | 
			
		||||
var documentmaxlength = config.documentmaxlength || 100000;
 | 
			
		||||
var documentmaxlength = config.documentmaxlength || 100000
 | 
			
		||||
 | 
			
		||||
// image upload setting, available options are imgur/s3/filesystem
 | 
			
		||||
var imageUploadType = process.env.HMD_IMAGE_UPLOAD_TYPE || config.imageUploadType || 'imgur';
 | 
			
		||||
var imageUploadType = process.env.HMD_IMAGE_UPLOAD_TYPE || config.imageUploadType || 'imgur'
 | 
			
		||||
 | 
			
		||||
config.s3 = config.s3 || {};
 | 
			
		||||
config.s3 = config.s3 || {}
 | 
			
		||||
var s3 = {
 | 
			
		||||
    accessKeyId: handleDockerSecret('s3_acccessKeyId') || process.env.HMD_S3_ACCESS_KEY_ID || config.s3.accessKeyId,
 | 
			
		||||
    secretAccessKey: handleDockerSecret('s3_secretAccessKey') || process.env.HMD_S3_SECRET_ACCESS_KEY || config.s3.secretAccessKey,
 | 
			
		||||
    region: process.env.HMD_S3_REGION || config.s3.region
 | 
			
		||||
  accessKeyId: handleDockerSecret('s3_acccessKeyId') || process.env.HMD_S3_ACCESS_KEY_ID || config.s3.accessKeyId,
 | 
			
		||||
  secretAccessKey: handleDockerSecret('s3_secretAccessKey') || process.env.HMD_S3_SECRET_ACCESS_KEY || config.s3.secretAccessKey,
 | 
			
		||||
  region: process.env.HMD_S3_REGION || config.s3.region
 | 
			
		||||
}
 | 
			
		||||
var s3bucket = process.env.HMD_S3_BUCKET || config.s3.bucket;
 | 
			
		||||
var s3bucket = process.env.HMD_S3_BUCKET || config.s3.bucket
 | 
			
		||||
 | 
			
		||||
// auth
 | 
			
		||||
var facebook = (process.env.HMD_FACEBOOK_CLIENTID && process.env.HMD_FACEBOOK_CLIENTSECRET || fs.existsSync('/run/secrets/facebook_clientID') && fs.existsSync('/run/secrets/facebook_clientSecret')) ? {
 | 
			
		||||
    clientID: handleDockerSecret('facebook_clientID') || process.env.HMD_FACEBOOK_CLIENTID,
 | 
			
		||||
    clientSecret: handleDockerSecret('facebook_clientSecret') || process.env.HMD_FACEBOOK_CLIENTSECRET
 | 
			
		||||
} : config.facebook || false;
 | 
			
		||||
var twitter = (process.env.HMD_TWITTER_CONSUMERKEY && process.env.HMD_TWITTER_CONSUMERSECRET || fs.existsSync('/run/secrets/twitter_consumerKey') && fs.existsSync('/run/secrets/twitter_consumerSecret')) ? {
 | 
			
		||||
    consumerKey: handleDockerSecret('twitter_consumerKey') || process.env.HMD_TWITTER_CONSUMERKEY,
 | 
			
		||||
    consumerSecret: handleDockerSecret('twitter_consumerSecret') || process.env.HMD_TWITTER_CONSUMERSECRET
 | 
			
		||||
} : config.twitter || false;
 | 
			
		||||
var github = (process.env.HMD_GITHUB_CLIENTID && process.env.HMD_GITHUB_CLIENTSECRET || fs.existsSync('/run/secrets/github_clientID') && fs.existsSync('/run/secrets/github_clientSecret')) ? {
 | 
			
		||||
    clientID: handleDockerSecret('github_clientID') || process.env.HMD_GITHUB_CLIENTID,
 | 
			
		||||
    clientSecret: handleDockerSecret('github_clientSecret') || process.env.HMD_GITHUB_CLIENTSECRET
 | 
			
		||||
} : config.github || false;
 | 
			
		||||
var gitlab = (process.env.HMD_GITLAB_CLIENTID && process.env.HMD_GITLAB_CLIENTSECRET || fs.existsSync('/run/secrets/gitlab_clientID') && fs.existsSync('/run/secrets/gitlab_clientSecret')) ? {
 | 
			
		||||
    baseURL: process.env.HMD_GITLAB_BASEURL,
 | 
			
		||||
    clientID: handleDockerSecret('gitlab_clientID') || process.env.HMD_GITLAB_CLIENTID,
 | 
			
		||||
    clientSecret: handleDockerSecret('gitlab_clientSecret') || process.env.HMD_GITLAB_CLIENTSECRET
 | 
			
		||||
} : config.gitlab || false;
 | 
			
		||||
var facebook = ((process.env.HMD_FACEBOOK_CLIENTID && process.env.HMD_FACEBOOK_CLIENTSECRET) || (fs.existsSync('/run/secrets/facebook_clientID') && fs.existsSync('/run/secrets/facebook_clientSecret'))) ? {
 | 
			
		||||
  clientID: handleDockerSecret('facebook_clientID') || process.env.HMD_FACEBOOK_CLIENTID,
 | 
			
		||||
  clientSecret: handleDockerSecret('facebook_clientSecret') || process.env.HMD_FACEBOOK_CLIENTSECRET
 | 
			
		||||
} : config.facebook || false
 | 
			
		||||
var twitter = ((process.env.HMD_TWITTER_CONSUMERKEY && process.env.HMD_TWITTER_CONSUMERSECRET) || (fs.existsSync('/run/secrets/twitter_consumerKey') && fs.existsSync('/run/secrets/twitter_consumerSecret'))) ? {
 | 
			
		||||
  consumerKey: handleDockerSecret('twitter_consumerKey') || process.env.HMD_TWITTER_CONSUMERKEY,
 | 
			
		||||
  consumerSecret: handleDockerSecret('twitter_consumerSecret') || process.env.HMD_TWITTER_CONSUMERSECRET
 | 
			
		||||
} : config.twitter || false
 | 
			
		||||
var github = ((process.env.HMD_GITHUB_CLIENTID && process.env.HMD_GITHUB_CLIENTSECRET) || (fs.existsSync('/run/secrets/github_clientID') && fs.existsSync('/run/secrets/github_clientSecret'))) ? {
 | 
			
		||||
  clientID: handleDockerSecret('github_clientID') || process.env.HMD_GITHUB_CLIENTID,
 | 
			
		||||
  clientSecret: handleDockerSecret('github_clientSecret') || process.env.HMD_GITHUB_CLIENTSECRET
 | 
			
		||||
} : config.github || false
 | 
			
		||||
var gitlab = ((process.env.HMD_GITLAB_CLIENTID && process.env.HMD_GITLAB_CLIENTSECRET) || (fs.existsSync('/run/secrets/gitlab_clientID') && fs.existsSync('/run/secrets/gitlab_clientSecret'))) ? {
 | 
			
		||||
  baseURL: process.env.HMD_GITLAB_BASEURL,
 | 
			
		||||
  clientID: handleDockerSecret('gitlab_clientID') || process.env.HMD_GITLAB_CLIENTID,
 | 
			
		||||
  clientSecret: handleDockerSecret('gitlab_clientSecret') || process.env.HMD_GITLAB_CLIENTSECRET
 | 
			
		||||
} : config.gitlab || false
 | 
			
		||||
var dropbox = ((process.env.HMD_DROPBOX_CLIENTID && process.env.HMD_DROPBOX_CLIENTSECRET) || (fs.existsSync('/run/secrets/dropbox_clientID') && fs.existsSync('/run/secrets/dropbox_clientSecret'))) ? {
 | 
			
		||||
    clientID: handleDockerSecret('dropbox_clientID') || process.env.HMD_DROPBOX_CLIENTID,
 | 
			
		||||
    clientSecret: handleDockerSecret('dropbox_clientSecret') || process.env.HMD_DROPBOX_CLIENTSECRET
 | 
			
		||||
} : (config.dropbox && config.dropbox.clientID && config.dropbox.clientSecret && config.dropbox) || false;
 | 
			
		||||
var google = ((process.env.HMD_GOOGLE_CLIENTID && process.env.HMD_GOOGLE_CLIENTSECRET)
 | 
			
		||||
              || (fs.existsSync('/run/secrets/google_clientID') && fs.existsSync('/run/secrets/google_clientSecret'))) ? {
 | 
			
		||||
    clientID: handleDockerSecret('google_clientID') || process.env.HMD_GOOGLE_CLIENTID,
 | 
			
		||||
    clientSecret: handleDockerSecret('google_clientSecret') || process.env.HMD_GOOGLE_CLIENTSECRET
 | 
			
		||||
} : (config.google && config.google.clientID && config.google.clientSecret && config.google) || false;
 | 
			
		||||
  clientID: handleDockerSecret('dropbox_clientID') || process.env.HMD_DROPBOX_CLIENTID,
 | 
			
		||||
  clientSecret: handleDockerSecret('dropbox_clientSecret') || process.env.HMD_DROPBOX_CLIENTSECRET
 | 
			
		||||
} : (config.dropbox && config.dropbox.clientID && config.dropbox.clientSecret && config.dropbox) || false
 | 
			
		||||
var google = ((process.env.HMD_GOOGLE_CLIENTID && process.env.HMD_GOOGLE_CLIENTSECRET) ||
 | 
			
		||||
              (fs.existsSync('/run/secrets/google_clientID') && fs.existsSync('/run/secrets/google_clientSecret'))) ? {
 | 
			
		||||
                clientID: handleDockerSecret('google_clientID') || process.env.HMD_GOOGLE_CLIENTID,
 | 
			
		||||
                clientSecret: handleDockerSecret('google_clientSecret') || process.env.HMD_GOOGLE_CLIENTSECRET
 | 
			
		||||
              } : (config.google && config.google.clientID && config.google.clientSecret && config.google) || false
 | 
			
		||||
var ldap = config.ldap || ((
 | 
			
		||||
    process.env.HMD_LDAP_URL ||
 | 
			
		||||
    process.env.HMD_LDAP_BINDDN ||
 | 
			
		||||
@ -123,106 +122,97 @@ var ldap = config.ldap || ((
 | 
			
		||||
    process.env.HMD_LDAP_SEARCHATTRIBUTES ||
 | 
			
		||||
    process.env.HMD_LDAP_TLS_CA ||
 | 
			
		||||
    process.env.HMD_LDAP_PROVIDERNAME
 | 
			
		||||
) ? {} : false);
 | 
			
		||||
if (process.env.HMD_LDAP_URL)
 | 
			
		||||
    ldap.url = process.env.HMD_LDAP_URL;
 | 
			
		||||
if (process.env.HMD_LDAP_BINDDN)
 | 
			
		||||
    ldap.bindDn = process.env.HMD_LDAP_BINDDN;
 | 
			
		||||
if (process.env.HMD_LDAP_BINDCREDENTIALS)
 | 
			
		||||
    ldap.bindCredentials = process.env.HMD_LDAP_BINDCREDENTIALS;
 | 
			
		||||
if (process.env.HMD_LDAP_TOKENSECRET)
 | 
			
		||||
    ldap.tokenSecret = process.env.HMD_LDAP_TOKENSECRET;
 | 
			
		||||
if (process.env.HMD_LDAP_SEARCHBASE)
 | 
			
		||||
    ldap.searchBase = process.env.HMD_LDAP_SEARCHBASE;
 | 
			
		||||
if (process.env.HMD_LDAP_SEARCHFILTER)
 | 
			
		||||
    ldap.searchFilter = process.env.HMD_LDAP_SEARCHFILTER;
 | 
			
		||||
if (process.env.HMD_LDAP_SEARCHATTRIBUTES)
 | 
			
		||||
    ldap.searchAttributes = process.env.HMD_LDAP_SEARCHATTRIBUTES;
 | 
			
		||||
) ? {} : false)
 | 
			
		||||
if (process.env.HMD_LDAP_URL) { ldap.url = process.env.HMD_LDAP_URL }
 | 
			
		||||
if (process.env.HMD_LDAP_BINDDN) { ldap.bindDn = process.env.HMD_LDAP_BINDDN }
 | 
			
		||||
if (process.env.HMD_LDAP_BINDCREDENTIALS) { ldap.bindCredentials = process.env.HMD_LDAP_BINDCREDENTIALS }
 | 
			
		||||
if (process.env.HMD_LDAP_TOKENSECRET) { ldap.tokenSecret = process.env.HMD_LDAP_TOKENSECRET }
 | 
			
		||||
if (process.env.HMD_LDAP_SEARCHBASE) { ldap.searchBase = process.env.HMD_LDAP_SEARCHBASE }
 | 
			
		||||
if (process.env.HMD_LDAP_SEARCHFILTER) { ldap.searchFilter = process.env.HMD_LDAP_SEARCHFILTER }
 | 
			
		||||
if (process.env.HMD_LDAP_SEARCHATTRIBUTES) { ldap.searchAttributes = process.env.HMD_LDAP_SEARCHATTRIBUTES }
 | 
			
		||||
if (process.env.HMD_LDAP_TLS_CA) {
 | 
			
		||||
    var ca = {
 | 
			
		||||
        ca: process.env.HMD_LDAP_TLS_CA.split(',')
 | 
			
		||||
    }
 | 
			
		||||
    ldap.tlsOptions = ldap.tlsOptions ? Object.assign(ldap.tlsOptions, ca) : ca;
 | 
			
		||||
    if (Array.isArray(ldap.tlsOptions.ca) && ldap.tlsOptions.ca.length > 0) {
 | 
			
		||||
        var i, len, results;
 | 
			
		||||
        results = [];
 | 
			
		||||
        for (i = 0, len = ldap.tlsOptions.ca.length; i < len; i++) {
 | 
			
		||||
            results.push(fs.readFileSync(ldap.tlsOptions.ca[i], 'utf8'));
 | 
			
		||||
        }
 | 
			
		||||
        ldap.tlsOptions.ca = results;
 | 
			
		||||
  var ca = {
 | 
			
		||||
    ca: process.env.HMD_LDAP_TLS_CA.split(',')
 | 
			
		||||
  }
 | 
			
		||||
  ldap.tlsOptions = ldap.tlsOptions ? Object.assign(ldap.tlsOptions, ca) : ca
 | 
			
		||||
  if (Array.isArray(ldap.tlsOptions.ca) && ldap.tlsOptions.ca.length > 0) {
 | 
			
		||||
    var i, len, results
 | 
			
		||||
    results = []
 | 
			
		||||
    for (i = 0, len = ldap.tlsOptions.ca.length; i < len; i++) {
 | 
			
		||||
      results.push(fs.readFileSync(ldap.tlsOptions.ca[i], 'utf8'))
 | 
			
		||||
    }
 | 
			
		||||
    ldap.tlsOptions.ca = results
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
if (process.env.HMD_LDAP_PROVIDERNAME) {
 | 
			
		||||
    ldap.providerName = process.env.HMD_LDAP_PROVIDERNAME;
 | 
			
		||||
  ldap.providerName = process.env.HMD_LDAP_PROVIDERNAME
 | 
			
		||||
}
 | 
			
		||||
var imgur = handleDockerSecret('imgur_clientid') || process.env.HMD_IMGUR_CLIENTID || config.imgur || false;
 | 
			
		||||
var email = process.env.HMD_EMAIL ? (process.env.HMD_EMAIL === 'true') : !!config.email;
 | 
			
		||||
var allowemailregister = process.env.HMD_ALLOW_EMAIL_REGISTER ? (process.env.HMD_ALLOW_EMAIL_REGISTER === 'true') : ((typeof config.allowemailregister === 'boolean') ? config.allowemailregister : true);
 | 
			
		||||
var imgur = handleDockerSecret('imgur_clientid') || process.env.HMD_IMGUR_CLIENTID || config.imgur || false
 | 
			
		||||
var email = process.env.HMD_EMAIL ? (process.env.HMD_EMAIL === 'true') : !!config.email
 | 
			
		||||
var allowemailregister = process.env.HMD_ALLOW_EMAIL_REGISTER ? (process.env.HMD_ALLOW_EMAIL_REGISTER === 'true') : ((typeof config.allowemailregister === 'boolean') ? config.allowemailregister : true)
 | 
			
		||||
 | 
			
		||||
function getserverurl() {
 | 
			
		||||
    var url = '';
 | 
			
		||||
    if (domain) {
 | 
			
		||||
        var protocol = protocolusessl ? 'https://' : 'http://';
 | 
			
		||||
        url = protocol + domain;
 | 
			
		||||
        if (urladdport && ((usessl && port != 443) || (!usessl && port != 80)))
 | 
			
		||||
            url += ':' + port;
 | 
			
		||||
    }
 | 
			
		||||
    if (urlpath)
 | 
			
		||||
        url += '/' + urlpath;
 | 
			
		||||
    return url;
 | 
			
		||||
function getserverurl () {
 | 
			
		||||
  var url = ''
 | 
			
		||||
  if (domain) {
 | 
			
		||||
    var protocol = protocolusessl ? 'https://' : 'http://'
 | 
			
		||||
    url = protocol + domain
 | 
			
		||||
    if (urladdport && ((usessl && port !== 443) || (!usessl && port !== 80))) { url += ':' + port }
 | 
			
		||||
  }
 | 
			
		||||
  if (urlpath) { url += '/' + urlpath }
 | 
			
		||||
  return url
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
var version = '0.5.0';
 | 
			
		||||
var minimumCompatibleVersion = '0.5.0';
 | 
			
		||||
var maintenance = true;
 | 
			
		||||
var cwd = path.join(__dirname, '..');
 | 
			
		||||
var version = '0.5.0'
 | 
			
		||||
var minimumCompatibleVersion = '0.5.0'
 | 
			
		||||
var maintenance = true
 | 
			
		||||
var cwd = path.join(__dirname, '..')
 | 
			
		||||
 | 
			
		||||
module.exports = {
 | 
			
		||||
    version: version,
 | 
			
		||||
    minimumCompatibleVersion: minimumCompatibleVersion,
 | 
			
		||||
    maintenance: maintenance,
 | 
			
		||||
    debug: debug,
 | 
			
		||||
    urlpath: urlpath,
 | 
			
		||||
    port: port,
 | 
			
		||||
    alloworigin: alloworigin,
 | 
			
		||||
    usessl: usessl,
 | 
			
		||||
    serverurl: getserverurl(),
 | 
			
		||||
    usecdn: usecdn,
 | 
			
		||||
    allowanonymous: allowanonymous,
 | 
			
		||||
    allowfreeurl: allowfreeurl,
 | 
			
		||||
    defaultpermission: defaultpermission,
 | 
			
		||||
    dburl: dburl,
 | 
			
		||||
    db: db,
 | 
			
		||||
    sslkeypath: path.join(cwd, sslkeypath),
 | 
			
		||||
    sslcertpath: path.join(cwd, sslcertpath),
 | 
			
		||||
    sslcapath: path.join(cwd, sslcapath),
 | 
			
		||||
    dhparampath: path.join(cwd, dhparampath),
 | 
			
		||||
    tmppath: path.join(cwd, tmppath),
 | 
			
		||||
    defaultnotepath: path.join(cwd, defaultnotepath),
 | 
			
		||||
    docspath: path.join(cwd, docspath),
 | 
			
		||||
    indexpath: path.join(cwd, indexpath),
 | 
			
		||||
    hackmdpath: path.join(cwd, hackmdpath),
 | 
			
		||||
    errorpath: path.join(cwd, errorpath),
 | 
			
		||||
    prettypath: path.join(cwd, prettypath),
 | 
			
		||||
    slidepath: path.join(cwd, slidepath),
 | 
			
		||||
    sessionname: sessionname,
 | 
			
		||||
    sessionsecret: sessionsecret,
 | 
			
		||||
    sessionlife: sessionlife,
 | 
			
		||||
    staticcachetime: staticcachetime,
 | 
			
		||||
    heartbeatinterval: heartbeatinterval,
 | 
			
		||||
    heartbeattimeout: heartbeattimeout,
 | 
			
		||||
    documentmaxlength: documentmaxlength,
 | 
			
		||||
    facebook: facebook,
 | 
			
		||||
    twitter: twitter,
 | 
			
		||||
    github: github,
 | 
			
		||||
    gitlab: gitlab,
 | 
			
		||||
    dropbox: dropbox,
 | 
			
		||||
    google: google,
 | 
			
		||||
    ldap: ldap,
 | 
			
		||||
    imgur: imgur,
 | 
			
		||||
    email: email,
 | 
			
		||||
    allowemailregister: allowemailregister,
 | 
			
		||||
    imageUploadType: imageUploadType,
 | 
			
		||||
    s3: s3,
 | 
			
		||||
    s3bucket: s3bucket
 | 
			
		||||
};
 | 
			
		||||
  version: version,
 | 
			
		||||
  minimumCompatibleVersion: minimumCompatibleVersion,
 | 
			
		||||
  maintenance: maintenance,
 | 
			
		||||
  debug: debug,
 | 
			
		||||
  urlpath: urlpath,
 | 
			
		||||
  port: port,
 | 
			
		||||
  alloworigin: alloworigin,
 | 
			
		||||
  usessl: usessl,
 | 
			
		||||
  serverurl: getserverurl(),
 | 
			
		||||
  usecdn: usecdn,
 | 
			
		||||
  allowanonymous: allowanonymous,
 | 
			
		||||
  allowfreeurl: allowfreeurl,
 | 
			
		||||
  defaultpermission: defaultpermission,
 | 
			
		||||
  dburl: dburl,
 | 
			
		||||
  db: db,
 | 
			
		||||
  sslkeypath: path.join(cwd, sslkeypath),
 | 
			
		||||
  sslcertpath: path.join(cwd, sslcertpath),
 | 
			
		||||
  sslcapath: path.join(cwd, sslcapath),
 | 
			
		||||
  dhparampath: path.join(cwd, dhparampath),
 | 
			
		||||
  tmppath: path.join(cwd, tmppath),
 | 
			
		||||
  defaultnotepath: path.join(cwd, defaultnotepath),
 | 
			
		||||
  docspath: path.join(cwd, docspath),
 | 
			
		||||
  indexpath: path.join(cwd, indexpath),
 | 
			
		||||
  hackmdpath: path.join(cwd, hackmdpath),
 | 
			
		||||
  errorpath: path.join(cwd, errorpath),
 | 
			
		||||
  prettypath: path.join(cwd, prettypath),
 | 
			
		||||
  slidepath: path.join(cwd, slidepath),
 | 
			
		||||
  sessionname: sessionname,
 | 
			
		||||
  sessionsecret: sessionsecret,
 | 
			
		||||
  sessionlife: sessionlife,
 | 
			
		||||
  staticcachetime: staticcachetime,
 | 
			
		||||
  heartbeatinterval: heartbeatinterval,
 | 
			
		||||
  heartbeattimeout: heartbeattimeout,
 | 
			
		||||
  documentmaxlength: documentmaxlength,
 | 
			
		||||
  facebook: facebook,
 | 
			
		||||
  twitter: twitter,
 | 
			
		||||
  github: github,
 | 
			
		||||
  gitlab: gitlab,
 | 
			
		||||
  dropbox: dropbox,
 | 
			
		||||
  google: google,
 | 
			
		||||
  ldap: ldap,
 | 
			
		||||
  imgur: imgur,
 | 
			
		||||
  email: email,
 | 
			
		||||
  allowemailregister: allowemailregister,
 | 
			
		||||
  imageUploadType: imageUploadType,
 | 
			
		||||
  s3: s3,
 | 
			
		||||
  s3bucket: s3bucket
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										309
									
								
								lib/history.js
									
									
									
									
									
								
							
							
						
						
									
										309
									
								
								lib/history.js
									
									
									
									
									
								
							@ -1,172 +1,175 @@
 | 
			
		||||
//history
 | 
			
		||||
//external modules
 | 
			
		||||
var async = require('async');
 | 
			
		||||
// history
 | 
			
		||||
// external modules
 | 
			
		||||
 | 
			
		||||
//core
 | 
			
		||||
var config = require("./config.js");
 | 
			
		||||
var logger = require("./logger.js");
 | 
			
		||||
var response = require("./response.js");
 | 
			
		||||
var models = require("./models");
 | 
			
		||||
// core
 | 
			
		||||
var config = require('./config.js')
 | 
			
		||||
var logger = require('./logger.js')
 | 
			
		||||
var response = require('./response.js')
 | 
			
		||||
var models = require('./models')
 | 
			
		||||
 | 
			
		||||
//public
 | 
			
		||||
// public
 | 
			
		||||
var History = {
 | 
			
		||||
    historyGet: historyGet,
 | 
			
		||||
    historyPost: historyPost,
 | 
			
		||||
    historyDelete: historyDelete,
 | 
			
		||||
    updateHistory: updateHistory
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
function getHistory(userid, callback) {
 | 
			
		||||
    models.User.findOne({
 | 
			
		||||
        where: {
 | 
			
		||||
            id: userid
 | 
			
		||||
        }
 | 
			
		||||
    }).then(function (user) {
 | 
			
		||||
        if (!user)
 | 
			
		||||
            return callback(null, null);
 | 
			
		||||
        var history = {};
 | 
			
		||||
        if (user.history)
 | 
			
		||||
            history = parseHistoryToObject(JSON.parse(user.history));
 | 
			
		||||
        if (config.debug)
 | 
			
		||||
            logger.info('read history success: ' + user.id);
 | 
			
		||||
        return callback(null, history);
 | 
			
		||||
    }).catch(function (err) {
 | 
			
		||||
        logger.error('read history failed: ' + err);
 | 
			
		||||
        return callback(err, null);
 | 
			
		||||
    });
 | 
			
		||||
  historyGet: historyGet,
 | 
			
		||||
  historyPost: historyPost,
 | 
			
		||||
  historyDelete: historyDelete,
 | 
			
		||||
  updateHistory: updateHistory
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function setHistory(userid, history, callback) {
 | 
			
		||||
    models.User.update({
 | 
			
		||||
        history: JSON.stringify(parseHistoryToArray(history))
 | 
			
		||||
    }, {
 | 
			
		||||
        where: {
 | 
			
		||||
            id: userid
 | 
			
		||||
        }
 | 
			
		||||
    }).then(function (count) {
 | 
			
		||||
        return callback(null, count);
 | 
			
		||||
    }).catch(function (err) {
 | 
			
		||||
        logger.error('set history failed: ' + err);
 | 
			
		||||
        return callback(err, null);
 | 
			
		||||
    });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function updateHistory(userid, noteId, document, time) {
 | 
			
		||||
    if (userid && noteId && typeof document !== 'undefined') {
 | 
			
		||||
        getHistory(userid, function (err, history) {
 | 
			
		||||
            if (err || !history) return;
 | 
			
		||||
            if (!history[noteId]) {
 | 
			
		||||
                history[noteId] = {};
 | 
			
		||||
            }
 | 
			
		||||
            var noteHistory = history[noteId];
 | 
			
		||||
            var noteInfo = models.Note.parseNoteInfo(document);
 | 
			
		||||
            noteHistory.id = noteId;
 | 
			
		||||
            noteHistory.text = noteInfo.title;
 | 
			
		||||
            noteHistory.time = time || Date.now();
 | 
			
		||||
            noteHistory.tags = noteInfo.tags;
 | 
			
		||||
            setHistory(userid, history, function (err, count) {
 | 
			
		||||
                return;
 | 
			
		||||
            });
 | 
			
		||||
        });
 | 
			
		||||
function getHistory (userid, callback) {
 | 
			
		||||
  models.User.findOne({
 | 
			
		||||
    where: {
 | 
			
		||||
      id: userid
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function parseHistoryToArray(history) {
 | 
			
		||||
    var _history = [];
 | 
			
		||||
    Object.keys(history).forEach(function (key) {
 | 
			
		||||
        var item = history[key];
 | 
			
		||||
        _history.push(item);
 | 
			
		||||
    });
 | 
			
		||||
    return _history;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function parseHistoryToObject(history) {
 | 
			
		||||
    var _history = {};
 | 
			
		||||
    for (var i = 0, l = history.length; i < l; i++) {
 | 
			
		||||
        var item = history[i];
 | 
			
		||||
        _history[item.id] = item;
 | 
			
		||||
  }).then(function (user) {
 | 
			
		||||
    if (!user) {
 | 
			
		||||
      return callback(null, null)
 | 
			
		||||
    }
 | 
			
		||||
    return _history;
 | 
			
		||||
    var history = {}
 | 
			
		||||
    if (user.history) {
 | 
			
		||||
      history = parseHistoryToObject(JSON.parse(user.history))
 | 
			
		||||
    }
 | 
			
		||||
    if (config.debug) {
 | 
			
		||||
      logger.info('read history success: ' + user.id)
 | 
			
		||||
    }
 | 
			
		||||
    return callback(null, history)
 | 
			
		||||
  }).catch(function (err) {
 | 
			
		||||
    logger.error('read history failed: ' + err)
 | 
			
		||||
    return callback(err, null)
 | 
			
		||||
  })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function historyGet(req, res) {
 | 
			
		||||
    if (req.isAuthenticated()) {
 | 
			
		||||
        getHistory(req.user.id, function (err, history) {
 | 
			
		||||
            if (err) return response.errorInternalError(res);
 | 
			
		||||
            if (!history) return response.errorNotFound(res);
 | 
			
		||||
            res.send({
 | 
			
		||||
                history: parseHistoryToArray(history)
 | 
			
		||||
            });
 | 
			
		||||
        });
 | 
			
		||||
function setHistory (userid, history, callback) {
 | 
			
		||||
  models.User.update({
 | 
			
		||||
    history: JSON.stringify(parseHistoryToArray(history))
 | 
			
		||||
  }, {
 | 
			
		||||
    where: {
 | 
			
		||||
      id: userid
 | 
			
		||||
    }
 | 
			
		||||
  }).then(function (count) {
 | 
			
		||||
    return callback(null, count)
 | 
			
		||||
  }).catch(function (err) {
 | 
			
		||||
    logger.error('set history failed: ' + err)
 | 
			
		||||
    return callback(err, null)
 | 
			
		||||
  })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function updateHistory (userid, noteId, document, time) {
 | 
			
		||||
  if (userid && noteId && typeof document !== 'undefined') {
 | 
			
		||||
    getHistory(userid, function (err, history) {
 | 
			
		||||
      if (err || !history) return
 | 
			
		||||
      if (!history[noteId]) {
 | 
			
		||||
        history[noteId] = {}
 | 
			
		||||
      }
 | 
			
		||||
      var noteHistory = history[noteId]
 | 
			
		||||
      var noteInfo = models.Note.parseNoteInfo(document)
 | 
			
		||||
      noteHistory.id = noteId
 | 
			
		||||
      noteHistory.text = noteInfo.title
 | 
			
		||||
      noteHistory.time = time || Date.now()
 | 
			
		||||
      noteHistory.tags = noteInfo.tags
 | 
			
		||||
      setHistory(userid, history, function (err, count) {
 | 
			
		||||
        if (err) {
 | 
			
		||||
          logger.log(err)
 | 
			
		||||
        }
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function parseHistoryToArray (history) {
 | 
			
		||||
  var _history = []
 | 
			
		||||
  Object.keys(history).forEach(function (key) {
 | 
			
		||||
    var item = history[key]
 | 
			
		||||
    _history.push(item)
 | 
			
		||||
  })
 | 
			
		||||
  return _history
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function parseHistoryToObject (history) {
 | 
			
		||||
  var _history = {}
 | 
			
		||||
  for (var i = 0, l = history.length; i < l; i++) {
 | 
			
		||||
    var item = history[i]
 | 
			
		||||
    _history[item.id] = item
 | 
			
		||||
  }
 | 
			
		||||
  return _history
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function historyGet (req, res) {
 | 
			
		||||
  if (req.isAuthenticated()) {
 | 
			
		||||
    getHistory(req.user.id, function (err, history) {
 | 
			
		||||
      if (err) return response.errorInternalError(res)
 | 
			
		||||
      if (!history) return response.errorNotFound(res)
 | 
			
		||||
      res.send({
 | 
			
		||||
        history: parseHistoryToArray(history)
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
  } else {
 | 
			
		||||
    return response.errorForbidden(res)
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function historyPost (req, res) {
 | 
			
		||||
  if (req.isAuthenticated()) {
 | 
			
		||||
    var noteId = req.params.noteId
 | 
			
		||||
    if (!noteId) {
 | 
			
		||||
      if (typeof req.body['history'] === 'undefined') return response.errorBadRequest(res)
 | 
			
		||||
      if (config.debug) { logger.info('SERVER received history from [' + req.user.id + ']: ' + req.body.history) }
 | 
			
		||||
      try {
 | 
			
		||||
        var history = JSON.parse(req.body.history)
 | 
			
		||||
      } catch (err) {
 | 
			
		||||
        return response.errorBadRequest(res)
 | 
			
		||||
      }
 | 
			
		||||
      if (Array.isArray(history)) {
 | 
			
		||||
        setHistory(req.user.id, history, function (err, count) {
 | 
			
		||||
          if (err) return response.errorInternalError(res)
 | 
			
		||||
          res.end()
 | 
			
		||||
        })
 | 
			
		||||
      } else {
 | 
			
		||||
        return response.errorBadRequest(res)
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
        return response.errorForbidden(res);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function historyPost(req, res) {
 | 
			
		||||
    if (req.isAuthenticated()) {
 | 
			
		||||
        var noteId = req.params.noteId;
 | 
			
		||||
        if (!noteId) {
 | 
			
		||||
            if (typeof req.body['history'] === 'undefined') return response.errorBadRequest(res);
 | 
			
		||||
            if (config.debug)
 | 
			
		||||
                logger.info('SERVER received history from [' + req.user.id + ']: ' + req.body.history);
 | 
			
		||||
            try {
 | 
			
		||||
                var history = JSON.parse(req.body.history);
 | 
			
		||||
            } catch (err) {
 | 
			
		||||
                return response.errorBadRequest(res);
 | 
			
		||||
            }
 | 
			
		||||
            if (Array.isArray(history)) {
 | 
			
		||||
                setHistory(req.user.id, history, function (err, count) {
 | 
			
		||||
                    if (err) return response.errorInternalError(res);
 | 
			
		||||
                    res.end();
 | 
			
		||||
                });
 | 
			
		||||
            } else {
 | 
			
		||||
                return response.errorBadRequest(res);
 | 
			
		||||
            }
 | 
			
		||||
      if (typeof req.body['pinned'] === 'undefined') return response.errorBadRequest(res)
 | 
			
		||||
      getHistory(req.user.id, function (err, history) {
 | 
			
		||||
        if (err) return response.errorInternalError(res)
 | 
			
		||||
        if (!history) return response.errorNotFound(res)
 | 
			
		||||
        if (!history[noteId]) return response.errorNotFound(res)
 | 
			
		||||
        if (req.body.pinned === 'true' || req.body.pinned === 'false') {
 | 
			
		||||
          history[noteId].pinned = (req.body.pinned === 'true')
 | 
			
		||||
          setHistory(req.user.id, history, function (err, count) {
 | 
			
		||||
            if (err) return response.errorInternalError(res)
 | 
			
		||||
            res.end()
 | 
			
		||||
          })
 | 
			
		||||
        } else {
 | 
			
		||||
            if (typeof req.body['pinned'] === 'undefined') return response.errorBadRequest(res);
 | 
			
		||||
            getHistory(req.user.id, function (err, history) {
 | 
			
		||||
                if (err) return response.errorInternalError(res);
 | 
			
		||||
                if (!history) return response.errorNotFound(res);
 | 
			
		||||
                if (!history[noteId]) return response.errorNotFound(res);
 | 
			
		||||
                if (req.body.pinned === 'true' || req.body.pinned === 'false') {
 | 
			
		||||
                    history[noteId].pinned = (req.body.pinned === 'true');
 | 
			
		||||
                    setHistory(req.user.id, history, function (err, count) {
 | 
			
		||||
                        if (err) return response.errorInternalError(res);
 | 
			
		||||
                        res.end();
 | 
			
		||||
                    });
 | 
			
		||||
                } else {
 | 
			
		||||
                    return response.errorBadRequest(res);
 | 
			
		||||
                }
 | 
			
		||||
            });
 | 
			
		||||
          return response.errorBadRequest(res)
 | 
			
		||||
        }
 | 
			
		||||
    } else {
 | 
			
		||||
        return response.errorForbidden(res);
 | 
			
		||||
      })
 | 
			
		||||
    }
 | 
			
		||||
  } else {
 | 
			
		||||
    return response.errorForbidden(res)
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function historyDelete(req, res) {
 | 
			
		||||
    if (req.isAuthenticated()) {
 | 
			
		||||
        var noteId = req.params.noteId;
 | 
			
		||||
        if (!noteId) {
 | 
			
		||||
            setHistory(req.user.id, [], function (err, count) {
 | 
			
		||||
                if (err) return response.errorInternalError(res);
 | 
			
		||||
                res.end();
 | 
			
		||||
            });
 | 
			
		||||
        } else {
 | 
			
		||||
            getHistory(req.user.id, function (err, history) {
 | 
			
		||||
                if (err) return response.errorInternalError(res);
 | 
			
		||||
                if (!history) return response.errorNotFound(res);
 | 
			
		||||
                delete history[noteId];
 | 
			
		||||
                setHistory(req.user.id, history, function (err, count) {
 | 
			
		||||
                    if (err) return response.errorInternalError(res);
 | 
			
		||||
                    res.end();
 | 
			
		||||
                });
 | 
			
		||||
            });
 | 
			
		||||
        }
 | 
			
		||||
function historyDelete (req, res) {
 | 
			
		||||
  if (req.isAuthenticated()) {
 | 
			
		||||
    var noteId = req.params.noteId
 | 
			
		||||
    if (!noteId) {
 | 
			
		||||
      setHistory(req.user.id, [], function (err, count) {
 | 
			
		||||
        if (err) return response.errorInternalError(res)
 | 
			
		||||
        res.end()
 | 
			
		||||
      })
 | 
			
		||||
    } else {
 | 
			
		||||
        return response.errorForbidden(res);
 | 
			
		||||
      getHistory(req.user.id, function (err, history) {
 | 
			
		||||
        if (err) return response.errorInternalError(res)
 | 
			
		||||
        if (!history) return response.errorNotFound(res)
 | 
			
		||||
        delete history[noteId]
 | 
			
		||||
        setHistory(req.user.id, history, function (err, count) {
 | 
			
		||||
          if (err) return response.errorInternalError(res)
 | 
			
		||||
          res.end()
 | 
			
		||||
        })
 | 
			
		||||
      })
 | 
			
		||||
    }
 | 
			
		||||
  } else {
 | 
			
		||||
    return response.errorForbidden(res)
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
module.exports = History;
 | 
			
		||||
module.exports = History
 | 
			
		||||
 | 
			
		||||
@ -1,25 +1,23 @@
 | 
			
		||||
"use strict";
 | 
			
		||||
 | 
			
		||||
// external modules
 | 
			
		||||
var randomcolor = require('randomcolor');
 | 
			
		||||
var randomcolor = require('randomcolor')
 | 
			
		||||
 | 
			
		||||
// core
 | 
			
		||||
module.exports = function(name) {
 | 
			
		||||
    var color = randomcolor({
 | 
			
		||||
        seed: name,
 | 
			
		||||
        luminosity: 'dark'
 | 
			
		||||
    });
 | 
			
		||||
    var letter = name.substring(0, 1).toUpperCase();
 | 
			
		||||
module.exports = function (name) {
 | 
			
		||||
  var color = randomcolor({
 | 
			
		||||
    seed: name,
 | 
			
		||||
    luminosity: 'dark'
 | 
			
		||||
  })
 | 
			
		||||
  var letter = name.substring(0, 1).toUpperCase()
 | 
			
		||||
 | 
			
		||||
    var svg = '<?xml version="1.0" encoding="UTF-8" standalone="no"?>';
 | 
			
		||||
    svg += '<svg xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns="http://www.w3.org/2000/svg" height="96" width="96" version="1.1" viewBox="0 0 96 96">';
 | 
			
		||||
    svg += '<g>';
 | 
			
		||||
    svg += '<rect width="96" height="96" fill="' + color + '" />';
 | 
			
		||||
    svg += '<text font-size="64px" font-family="sans-serif" text-anchor="middle" fill="#ffffff">';
 | 
			
		||||
    svg += '<tspan x="48" y="72" stroke-width=".26458px" fill="#ffffff">' + letter + '</tspan>';
 | 
			
		||||
    svg += '</text>';
 | 
			
		||||
    svg += '</g>';
 | 
			
		||||
    svg += '</svg>';
 | 
			
		||||
  var svg = '<?xml version="1.0" encoding="UTF-8" standalone="no"?>'
 | 
			
		||||
  svg += '<svg xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns="http://www.w3.org/2000/svg" height="96" width="96" version="1.1" viewBox="0 0 96 96">'
 | 
			
		||||
  svg += '<g>'
 | 
			
		||||
  svg += '<rect width="96" height="96" fill="' + color + '" />'
 | 
			
		||||
  svg += '<text font-size="64px" font-family="sans-serif" text-anchor="middle" fill="#ffffff">'
 | 
			
		||||
  svg += '<tspan x="48" y="72" stroke-width=".26458px" fill="#ffffff">' + letter + '</tspan>'
 | 
			
		||||
  svg += '</text>'
 | 
			
		||||
  svg += '</g>'
 | 
			
		||||
  svg += '</svg>'
 | 
			
		||||
 | 
			
		||||
    return 'data:image/svg+xml;base64,' + new Buffer(svg).toString('base64');
 | 
			
		||||
};
 | 
			
		||||
  return 'data:image/svg+xml;base64,' + new Buffer(svg).toString('base64')
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -1,22 +1,22 @@
 | 
			
		||||
var winston = require('winston');
 | 
			
		||||
winston.emitErrs = true;
 | 
			
		||||
var winston = require('winston')
 | 
			
		||||
winston.emitErrs = true
 | 
			
		||||
 | 
			
		||||
var logger = new winston.Logger({
 | 
			
		||||
    transports: [
 | 
			
		||||
        new winston.transports.Console({
 | 
			
		||||
            level: 'debug',
 | 
			
		||||
            handleExceptions: true,
 | 
			
		||||
            json: false,
 | 
			
		||||
            colorize: true,
 | 
			
		||||
            timestamp: true
 | 
			
		||||
        })
 | 
			
		||||
    ],
 | 
			
		||||
    exitOnError: false
 | 
			
		||||
});
 | 
			
		||||
  transports: [
 | 
			
		||||
    new winston.transports.Console({
 | 
			
		||||
      level: 'debug',
 | 
			
		||||
      handleExceptions: true,
 | 
			
		||||
      json: false,
 | 
			
		||||
      colorize: true,
 | 
			
		||||
      timestamp: true
 | 
			
		||||
    })
 | 
			
		||||
  ],
 | 
			
		||||
  exitOnError: false
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
module.exports = logger;
 | 
			
		||||
module.exports = logger
 | 
			
		||||
module.exports.stream = {
 | 
			
		||||
    write: function(message, encoding){
 | 
			
		||||
        logger.info(message);
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
  write: function (message, encoding) {
 | 
			
		||||
    logger.info(message)
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -1,15 +1,11 @@
 | 
			
		||||
"use strict";
 | 
			
		||||
 | 
			
		||||
module.exports = {
 | 
			
		||||
    up: function (queryInterface, Sequelize) {
 | 
			
		||||
        queryInterface.addColumn('Users', 'accessToken', Sequelize.STRING);
 | 
			
		||||
        queryInterface.addColumn('Users', 'refreshToken', Sequelize.STRING);
 | 
			
		||||
        return;
 | 
			
		||||
    },
 | 
			
		||||
  up: function (queryInterface, Sequelize) {
 | 
			
		||||
    queryInterface.addColumn('Users', 'accessToken', Sequelize.STRING)
 | 
			
		||||
    queryInterface.addColumn('Users', 'refreshToken', Sequelize.STRING)
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
    down: function (queryInterface, Sequelize) {
 | 
			
		||||
        queryInterface.removeColumn('Users', 'accessToken');
 | 
			
		||||
        queryInterface.removeColumn('Users', 'refreshToken');
 | 
			
		||||
        return;
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
  down: function (queryInterface, Sequelize) {
 | 
			
		||||
    queryInterface.removeColumn('Users', 'accessToken')
 | 
			
		||||
    queryInterface.removeColumn('Users', 'refreshToken')
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -1,8 +1,6 @@
 | 
			
		||||
'use strict';
 | 
			
		||||
 | 
			
		||||
module.exports = {
 | 
			
		||||
  up: function (queryInterface, Sequelize) {
 | 
			
		||||
    queryInterface.addColumn('Notes', 'savedAt', Sequelize.DATE);
 | 
			
		||||
    queryInterface.addColumn('Notes', 'savedAt', Sequelize.DATE)
 | 
			
		||||
    queryInterface.createTable('Revisions', {
 | 
			
		||||
      id: {
 | 
			
		||||
        type: Sequelize.UUID,
 | 
			
		||||
@ -15,13 +13,11 @@ module.exports = {
 | 
			
		||||
      length: Sequelize.INTEGER,
 | 
			
		||||
      createdAt: Sequelize.DATE,
 | 
			
		||||
      updatedAt: Sequelize.DATE
 | 
			
		||||
    });
 | 
			
		||||
    return;
 | 
			
		||||
    })
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  down: function (queryInterface, Sequelize) {
 | 
			
		||||
    queryInterface.dropTable('Revisions');
 | 
			
		||||
    queryInterface.removeColumn('Notes', 'savedAt');
 | 
			
		||||
    return;
 | 
			
		||||
    queryInterface.dropTable('Revisions')
 | 
			
		||||
    queryInterface.removeColumn('Notes', 'savedAt')
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -1,9 +1,7 @@
 | 
			
		||||
'use strict';
 | 
			
		||||
 | 
			
		||||
module.exports = {
 | 
			
		||||
  up: function (queryInterface, Sequelize) {
 | 
			
		||||
    queryInterface.addColumn('Notes', 'authorship', Sequelize.TEXT);
 | 
			
		||||
    queryInterface.addColumn('Revisions', 'authorship', Sequelize.TEXT);
 | 
			
		||||
    queryInterface.addColumn('Notes', 'authorship', Sequelize.TEXT)
 | 
			
		||||
    queryInterface.addColumn('Revisions', 'authorship', Sequelize.TEXT)
 | 
			
		||||
    queryInterface.createTable('Authors', {
 | 
			
		||||
      id: {
 | 
			
		||||
        type: Sequelize.INTEGER,
 | 
			
		||||
@ -15,14 +13,12 @@ module.exports = {
 | 
			
		||||
      userId: Sequelize.UUID,
 | 
			
		||||
      createdAt: Sequelize.DATE,
 | 
			
		||||
      updatedAt: Sequelize.DATE
 | 
			
		||||
    });
 | 
			
		||||
    return;
 | 
			
		||||
    })
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  down: function (queryInterface, Sequelize) {
 | 
			
		||||
    queryInterface.dropTable('Authors');
 | 
			
		||||
    queryInterface.removeColumn('Revisions', 'authorship');
 | 
			
		||||
    queryInterface.removeColumn('Notes', 'authorship');
 | 
			
		||||
    return;
 | 
			
		||||
    queryInterface.dropTable('Authors')
 | 
			
		||||
    queryInterface.removeColumn('Revisions', 'authorship')
 | 
			
		||||
    queryInterface.removeColumn('Notes', 'authorship')
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -1,11 +1,9 @@
 | 
			
		||||
'use strict';
 | 
			
		||||
 | 
			
		||||
module.exports = {
 | 
			
		||||
  up: function (queryInterface, Sequelize) {
 | 
			
		||||
    queryInterface.addColumn('Notes', 'deletedAt', Sequelize.DATE);
 | 
			
		||||
    queryInterface.addColumn('Notes', 'deletedAt', Sequelize.DATE)
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  down: function (queryInterface, Sequelize) {
 | 
			
		||||
    queryInterface.removeColumn('Notes', 'deletedAt');
 | 
			
		||||
    queryInterface.removeColumn('Notes', 'deletedAt')
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -1,13 +1,11 @@
 | 
			
		||||
'use strict';
 | 
			
		||||
 | 
			
		||||
module.exports = {
 | 
			
		||||
  up: function (queryInterface, Sequelize) {
 | 
			
		||||
    queryInterface.addColumn('Users', 'email', Sequelize.TEXT);
 | 
			
		||||
    queryInterface.addColumn('Users', 'password', Sequelize.TEXT);
 | 
			
		||||
    queryInterface.addColumn('Users', 'email', Sequelize.TEXT)
 | 
			
		||||
    queryInterface.addColumn('Users', 'password', Sequelize.TEXT)
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  down: function (queryInterface, Sequelize) {
 | 
			
		||||
    queryInterface.removeColumn('Users', 'email');
 | 
			
		||||
    queryInterface.removeColumn('Users', 'password');
 | 
			
		||||
    queryInterface.removeColumn('Users', 'email')
 | 
			
		||||
    queryInterface.removeColumn('Users', 'password')
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -1,43 +1,37 @@
 | 
			
		||||
"use strict";
 | 
			
		||||
 | 
			
		||||
// external modules
 | 
			
		||||
var Sequelize = require("sequelize");
 | 
			
		||||
 | 
			
		||||
// core
 | 
			
		||||
var logger = require("../logger.js");
 | 
			
		||||
var Sequelize = require('sequelize')
 | 
			
		||||
 | 
			
		||||
module.exports = function (sequelize, DataTypes) {
 | 
			
		||||
    var Author = sequelize.define("Author", {
 | 
			
		||||
        id: {
 | 
			
		||||
            type: Sequelize.INTEGER,
 | 
			
		||||
            primaryKey: true,
 | 
			
		||||
            autoIncrement: true
 | 
			
		||||
        },
 | 
			
		||||
        color: {
 | 
			
		||||
            type: DataTypes.STRING
 | 
			
		||||
        }
 | 
			
		||||
    }, {
 | 
			
		||||
        indexes: [
 | 
			
		||||
            {
 | 
			
		||||
                unique: true,
 | 
			
		||||
                fields: ['noteId', 'userId']
 | 
			
		||||
            }
 | 
			
		||||
        ],
 | 
			
		||||
        classMethods: {
 | 
			
		||||
            associate: function (models) {
 | 
			
		||||
                Author.belongsTo(models.Note, {
 | 
			
		||||
                    foreignKey: "noteId",
 | 
			
		||||
                    as: "note",
 | 
			
		||||
                    constraints: false
 | 
			
		||||
                });
 | 
			
		||||
                Author.belongsTo(models.User, {
 | 
			
		||||
                    foreignKey: "userId",
 | 
			
		||||
                    as: "user",
 | 
			
		||||
                    constraints: false
 | 
			
		||||
                });
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
    return Author;
 | 
			
		||||
};
 | 
			
		||||
  var Author = sequelize.define('Author', {
 | 
			
		||||
    id: {
 | 
			
		||||
      type: Sequelize.INTEGER,
 | 
			
		||||
      primaryKey: true,
 | 
			
		||||
      autoIncrement: true
 | 
			
		||||
    },
 | 
			
		||||
    color: {
 | 
			
		||||
      type: DataTypes.STRING
 | 
			
		||||
    }
 | 
			
		||||
  }, {
 | 
			
		||||
    indexes: [
 | 
			
		||||
      {
 | 
			
		||||
        unique: true,
 | 
			
		||||
        fields: ['noteId', 'userId']
 | 
			
		||||
      }
 | 
			
		||||
    ],
 | 
			
		||||
    classMethods: {
 | 
			
		||||
      associate: function (models) {
 | 
			
		||||
        Author.belongsTo(models.Note, {
 | 
			
		||||
          foreignKey: 'noteId',
 | 
			
		||||
          as: 'note',
 | 
			
		||||
          constraints: false
 | 
			
		||||
        })
 | 
			
		||||
        Author.belongsTo(models.User, {
 | 
			
		||||
          foreignKey: 'userId',
 | 
			
		||||
          as: 'user',
 | 
			
		||||
          constraints: false
 | 
			
		||||
        })
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  })
 | 
			
		||||
  return Author
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -1,57 +1,55 @@
 | 
			
		||||
"use strict";
 | 
			
		||||
 | 
			
		||||
// external modules
 | 
			
		||||
var fs = require("fs");
 | 
			
		||||
var path = require("path");
 | 
			
		||||
var Sequelize = require("sequelize");
 | 
			
		||||
var fs = require('fs')
 | 
			
		||||
var path = require('path')
 | 
			
		||||
var Sequelize = require('sequelize')
 | 
			
		||||
 | 
			
		||||
// core
 | 
			
		||||
var config = require('../config.js');
 | 
			
		||||
var logger = require("../logger.js");
 | 
			
		||||
var config = require('../config.js')
 | 
			
		||||
var logger = require('../logger.js')
 | 
			
		||||
 | 
			
		||||
var dbconfig = config.db;
 | 
			
		||||
dbconfig.logging = config.debug ? logger.info : false;
 | 
			
		||||
var dbconfig = config.db
 | 
			
		||||
dbconfig.logging = config.debug ? logger.info : false
 | 
			
		||||
 | 
			
		||||
var sequelize = null;
 | 
			
		||||
var sequelize = null
 | 
			
		||||
 | 
			
		||||
// Heroku specific
 | 
			
		||||
if (config.dburl)
 | 
			
		||||
    sequelize = new Sequelize(config.dburl, dbconfig);
 | 
			
		||||
else
 | 
			
		||||
    sequelize = new Sequelize(dbconfig.database, dbconfig.username, dbconfig.password, dbconfig);
 | 
			
		||||
if (config.dburl) {
 | 
			
		||||
  sequelize = new Sequelize(config.dburl, dbconfig)
 | 
			
		||||
} else {
 | 
			
		||||
  sequelize = new Sequelize(dbconfig.database, dbconfig.username, dbconfig.password, dbconfig)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// [Postgres] Handling NULL bytes
 | 
			
		||||
// https://github.com/sequelize/sequelize/issues/6485
 | 
			
		||||
function stripNullByte(value) {
 | 
			
		||||
    return value ? value.replace(/\u0000/g, "") : value;
 | 
			
		||||
function stripNullByte (value) {
 | 
			
		||||
  return value ? value.replace(/\u0000/g, '') : value
 | 
			
		||||
}
 | 
			
		||||
sequelize.stripNullByte = stripNullByte;
 | 
			
		||||
sequelize.stripNullByte = stripNullByte
 | 
			
		||||
 | 
			
		||||
function processData(data, _default, process) {
 | 
			
		||||
    if (data === undefined) return data;
 | 
			
		||||
    else return data === null ? _default : (process ? process(data) : data);
 | 
			
		||||
function processData (data, _default, process) {
 | 
			
		||||
  if (data === undefined) return data
 | 
			
		||||
  else return data === null ? _default : (process ? process(data) : data)
 | 
			
		||||
}
 | 
			
		||||
sequelize.processData = processData;
 | 
			
		||||
sequelize.processData = processData
 | 
			
		||||
 | 
			
		||||
var db = {};
 | 
			
		||||
var db = {}
 | 
			
		||||
 | 
			
		||||
fs
 | 
			
		||||
    .readdirSync(__dirname)
 | 
			
		||||
fs.readdirSync(__dirname)
 | 
			
		||||
    .filter(function (file) {
 | 
			
		||||
        return (file.indexOf(".") !== 0) && (file !== "index.js");
 | 
			
		||||
      return (file.indexOf('.') !== 0) && (file !== 'index.js')
 | 
			
		||||
    })
 | 
			
		||||
    .forEach(function (file) {
 | 
			
		||||
        var model = sequelize.import(path.join(__dirname, file));
 | 
			
		||||
        db[model.name] = model;
 | 
			
		||||
    });
 | 
			
		||||
      var model = sequelize.import(path.join(__dirname, file))
 | 
			
		||||
      db[model.name] = model
 | 
			
		||||
    })
 | 
			
		||||
 | 
			
		||||
Object.keys(db).forEach(function (modelName) {
 | 
			
		||||
    if ("associate" in db[modelName]) {
 | 
			
		||||
        db[modelName].associate(db);
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
  if ('associate' in db[modelName]) {
 | 
			
		||||
    db[modelName].associate(db)
 | 
			
		||||
  }
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
db.sequelize = sequelize;
 | 
			
		||||
db.Sequelize = Sequelize;
 | 
			
		||||
db.sequelize = sequelize
 | 
			
		||||
db.Sequelize = Sequelize
 | 
			
		||||
 | 
			
		||||
module.exports = db;
 | 
			
		||||
module.exports = db
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										1021
									
								
								lib/models/note.js
									
									
									
									
									
								
							
							
						
						
									
										1021
									
								
								lib/models/note.js
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							@ -1,306 +1,306 @@
 | 
			
		||||
"use strict";
 | 
			
		||||
 | 
			
		||||
// external modules
 | 
			
		||||
var Sequelize = require("sequelize");
 | 
			
		||||
var async = require('async');
 | 
			
		||||
var moment = require('moment');
 | 
			
		||||
var childProcess = require('child_process');
 | 
			
		||||
var shortId = require('shortid');
 | 
			
		||||
var Sequelize = require('sequelize')
 | 
			
		||||
var async = require('async')
 | 
			
		||||
var moment = require('moment')
 | 
			
		||||
var childProcess = require('child_process')
 | 
			
		||||
var shortId = require('shortid')
 | 
			
		||||
 | 
			
		||||
// core
 | 
			
		||||
var config = require("../config.js");
 | 
			
		||||
var logger = require("../logger.js");
 | 
			
		||||
var config = require('../config.js')
 | 
			
		||||
var logger = require('../logger.js')
 | 
			
		||||
 | 
			
		||||
var dmpWorker = createDmpWorker();
 | 
			
		||||
var dmpCallbackCache = {};
 | 
			
		||||
var dmpWorker = createDmpWorker()
 | 
			
		||||
var dmpCallbackCache = {}
 | 
			
		||||
 | 
			
		||||
function createDmpWorker() {
 | 
			
		||||
    var worker = childProcess.fork("./lib/workers/dmpWorker.js", {
 | 
			
		||||
        stdio: 'ignore'
 | 
			
		||||
    });
 | 
			
		||||
    if (config.debug) logger.info('dmp worker process started');
 | 
			
		||||
    worker.on('message', function (data) {
 | 
			
		||||
        if (!data || !data.msg || !data.cacheKey) {
 | 
			
		||||
            return logger.error('dmp worker error: not enough data on message');
 | 
			
		||||
        }
 | 
			
		||||
        var cacheKey = data.cacheKey;
 | 
			
		||||
        switch(data.msg) {
 | 
			
		||||
            case 'error':
 | 
			
		||||
                dmpCallbackCache[cacheKey](data.error, null);
 | 
			
		||||
                break;
 | 
			
		||||
            case 'check':
 | 
			
		||||
                dmpCallbackCache[cacheKey](null, data.result);
 | 
			
		||||
                break;
 | 
			
		||||
        }
 | 
			
		||||
        delete dmpCallbackCache[cacheKey];
 | 
			
		||||
    });
 | 
			
		||||
    worker.on('close', function (code) {
 | 
			
		||||
        dmpWorker = null;
 | 
			
		||||
        if (config.debug) logger.info('dmp worker process exited with code ' + code);
 | 
			
		||||
    });
 | 
			
		||||
    return worker;
 | 
			
		||||
function createDmpWorker () {
 | 
			
		||||
  var worker = childProcess.fork('./lib/workers/dmpWorker.js', {
 | 
			
		||||
    stdio: 'ignore'
 | 
			
		||||
  })
 | 
			
		||||
  if (config.debug) logger.info('dmp worker process started')
 | 
			
		||||
  worker.on('message', function (data) {
 | 
			
		||||
    if (!data || !data.msg || !data.cacheKey) {
 | 
			
		||||
      return logger.error('dmp worker error: not enough data on message')
 | 
			
		||||
    }
 | 
			
		||||
    var cacheKey = data.cacheKey
 | 
			
		||||
    switch (data.msg) {
 | 
			
		||||
      case 'error':
 | 
			
		||||
        dmpCallbackCache[cacheKey](data.error, null)
 | 
			
		||||
        break
 | 
			
		||||
      case 'check':
 | 
			
		||||
        dmpCallbackCache[cacheKey](null, data.result)
 | 
			
		||||
        break
 | 
			
		||||
    }
 | 
			
		||||
    delete dmpCallbackCache[cacheKey]
 | 
			
		||||
  })
 | 
			
		||||
  worker.on('close', function (code) {
 | 
			
		||||
    dmpWorker = null
 | 
			
		||||
    if (config.debug) logger.info('dmp worker process exited with code ' + code)
 | 
			
		||||
  })
 | 
			
		||||
  return worker
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function sendDmpWorker(data, callback) {
 | 
			
		||||
    if (!dmpWorker) dmpWorker = createDmpWorker();
 | 
			
		||||
    var cacheKey = Date.now() + '_' + shortId.generate();
 | 
			
		||||
    dmpCallbackCache[cacheKey] = callback;
 | 
			
		||||
    data = Object.assign(data, {
 | 
			
		||||
        cacheKey: cacheKey
 | 
			
		||||
    });
 | 
			
		||||
    dmpWorker.send(data);
 | 
			
		||||
function sendDmpWorker (data, callback) {
 | 
			
		||||
  if (!dmpWorker) dmpWorker = createDmpWorker()
 | 
			
		||||
  var cacheKey = Date.now() + '_' + shortId.generate()
 | 
			
		||||
  dmpCallbackCache[cacheKey] = callback
 | 
			
		||||
  data = Object.assign(data, {
 | 
			
		||||
    cacheKey: cacheKey
 | 
			
		||||
  })
 | 
			
		||||
  dmpWorker.send(data)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
module.exports = function (sequelize, DataTypes) {
 | 
			
		||||
    var Revision = sequelize.define("Revision", {
 | 
			
		||||
        id: {
 | 
			
		||||
            type: DataTypes.UUID,
 | 
			
		||||
            primaryKey: true,
 | 
			
		||||
            defaultValue: Sequelize.UUIDV4
 | 
			
		||||
        },
 | 
			
		||||
        patch: {
 | 
			
		||||
            type: DataTypes.TEXT,
 | 
			
		||||
            get: function () {
 | 
			
		||||
                return sequelize.processData(this.getDataValue('patch'), "");
 | 
			
		||||
  var Revision = sequelize.define('Revision', {
 | 
			
		||||
    id: {
 | 
			
		||||
      type: DataTypes.UUID,
 | 
			
		||||
      primaryKey: true,
 | 
			
		||||
      defaultValue: Sequelize.UUIDV4
 | 
			
		||||
    },
 | 
			
		||||
    patch: {
 | 
			
		||||
      type: DataTypes.TEXT,
 | 
			
		||||
      get: function () {
 | 
			
		||||
        return sequelize.processData(this.getDataValue('patch'), '')
 | 
			
		||||
      },
 | 
			
		||||
      set: function (value) {
 | 
			
		||||
        this.setDataValue('patch', sequelize.stripNullByte(value))
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    lastContent: {
 | 
			
		||||
      type: DataTypes.TEXT,
 | 
			
		||||
      get: function () {
 | 
			
		||||
        return sequelize.processData(this.getDataValue('lastContent'), '')
 | 
			
		||||
      },
 | 
			
		||||
      set: function (value) {
 | 
			
		||||
        this.setDataValue('lastContent', sequelize.stripNullByte(value))
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    content: {
 | 
			
		||||
      type: DataTypes.TEXT,
 | 
			
		||||
      get: function () {
 | 
			
		||||
        return sequelize.processData(this.getDataValue('content'), '')
 | 
			
		||||
      },
 | 
			
		||||
      set: function (value) {
 | 
			
		||||
        this.setDataValue('content', sequelize.stripNullByte(value))
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    length: {
 | 
			
		||||
      type: DataTypes.INTEGER
 | 
			
		||||
    },
 | 
			
		||||
    authorship: {
 | 
			
		||||
      type: DataTypes.TEXT,
 | 
			
		||||
      get: function () {
 | 
			
		||||
        return sequelize.processData(this.getDataValue('authorship'), [], JSON.parse)
 | 
			
		||||
      },
 | 
			
		||||
      set: function (value) {
 | 
			
		||||
        this.setDataValue('authorship', value ? JSON.stringify(value) : value)
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }, {
 | 
			
		||||
    classMethods: {
 | 
			
		||||
      associate: function (models) {
 | 
			
		||||
        Revision.belongsTo(models.Note, {
 | 
			
		||||
          foreignKey: 'noteId',
 | 
			
		||||
          as: 'note',
 | 
			
		||||
          constraints: false
 | 
			
		||||
        })
 | 
			
		||||
      },
 | 
			
		||||
      getNoteRevisions: function (note, callback) {
 | 
			
		||||
        Revision.findAll({
 | 
			
		||||
          where: {
 | 
			
		||||
            noteId: note.id
 | 
			
		||||
          },
 | 
			
		||||
          order: '"createdAt" DESC'
 | 
			
		||||
        }).then(function (revisions) {
 | 
			
		||||
          var data = []
 | 
			
		||||
          for (var i = 0, l = revisions.length; i < l; i++) {
 | 
			
		||||
            var revision = revisions[i]
 | 
			
		||||
            data.push({
 | 
			
		||||
              time: moment(revision.createdAt).valueOf(),
 | 
			
		||||
              length: revision.length
 | 
			
		||||
            })
 | 
			
		||||
          }
 | 
			
		||||
          callback(null, data)
 | 
			
		||||
        }).catch(function (err) {
 | 
			
		||||
          callback(err, null)
 | 
			
		||||
        })
 | 
			
		||||
      },
 | 
			
		||||
      getPatchedNoteRevisionByTime: function (note, time, callback) {
 | 
			
		||||
        // find all revisions to prepare for all possible calculation
 | 
			
		||||
        Revision.findAll({
 | 
			
		||||
          where: {
 | 
			
		||||
            noteId: note.id
 | 
			
		||||
          },
 | 
			
		||||
          order: '"createdAt" DESC'
 | 
			
		||||
        }).then(function (revisions) {
 | 
			
		||||
          if (revisions.length <= 0) return callback(null, null)
 | 
			
		||||
          // measure target revision position
 | 
			
		||||
          Revision.count({
 | 
			
		||||
            where: {
 | 
			
		||||
              noteId: note.id,
 | 
			
		||||
              createdAt: {
 | 
			
		||||
                $gte: time
 | 
			
		||||
              }
 | 
			
		||||
            },
 | 
			
		||||
            set: function (value) {
 | 
			
		||||
                this.setDataValue('patch', sequelize.stripNullByte(value));
 | 
			
		||||
            }
 | 
			
		||||
        },
 | 
			
		||||
        lastContent: {
 | 
			
		||||
            type: DataTypes.TEXT,
 | 
			
		||||
            get: function () {
 | 
			
		||||
                return sequelize.processData(this.getDataValue('lastContent'), "");
 | 
			
		||||
            },
 | 
			
		||||
            set: function (value) {
 | 
			
		||||
                this.setDataValue('lastContent', sequelize.stripNullByte(value));
 | 
			
		||||
            }
 | 
			
		||||
        },
 | 
			
		||||
        content: {
 | 
			
		||||
            type: DataTypes.TEXT,
 | 
			
		||||
            get: function () {
 | 
			
		||||
                return sequelize.processData(this.getDataValue('content'), "");
 | 
			
		||||
            },
 | 
			
		||||
            set: function (value) {
 | 
			
		||||
                this.setDataValue('content', sequelize.stripNullByte(value));
 | 
			
		||||
            }
 | 
			
		||||
        },
 | 
			
		||||
        length: {
 | 
			
		||||
            type: DataTypes.INTEGER
 | 
			
		||||
        },
 | 
			
		||||
        authorship: {
 | 
			
		||||
            type: DataTypes.TEXT,
 | 
			
		||||
            get: function () {
 | 
			
		||||
                return sequelize.processData(this.getDataValue('authorship'), [], JSON.parse);
 | 
			
		||||
            },
 | 
			
		||||
            set: function (value) {
 | 
			
		||||
                this.setDataValue('authorship', value ? JSON.stringify(value) : value);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }, {
 | 
			
		||||
        classMethods: {
 | 
			
		||||
            associate: function (models) {
 | 
			
		||||
                Revision.belongsTo(models.Note, {
 | 
			
		||||
                    foreignKey: "noteId",
 | 
			
		||||
                    as: "note",
 | 
			
		||||
                    constraints: false
 | 
			
		||||
                });
 | 
			
		||||
            },
 | 
			
		||||
            getNoteRevisions: function (note, callback) {
 | 
			
		||||
                Revision.findAll({
 | 
			
		||||
                    where: {
 | 
			
		||||
                        noteId: note.id
 | 
			
		||||
                    },
 | 
			
		||||
                    order: '"createdAt" DESC'
 | 
			
		||||
                }).then(function (revisions) {
 | 
			
		||||
                    var data = [];
 | 
			
		||||
                    for (var i = 0, l = revisions.length; i < l; i++) {
 | 
			
		||||
                        var revision = revisions[i];
 | 
			
		||||
                        data.push({
 | 
			
		||||
                            time: moment(revision.createdAt).valueOf(),
 | 
			
		||||
                            length: revision.length
 | 
			
		||||
                        });
 | 
			
		||||
            order: '"createdAt" DESC'
 | 
			
		||||
          }).then(function (count) {
 | 
			
		||||
            if (count <= 0) return callback(null, null)
 | 
			
		||||
            sendDmpWorker({
 | 
			
		||||
              msg: 'get revision',
 | 
			
		||||
              revisions: revisions,
 | 
			
		||||
              count: count
 | 
			
		||||
            }, callback)
 | 
			
		||||
          }).catch(function (err) {
 | 
			
		||||
            return callback(err, null)
 | 
			
		||||
          })
 | 
			
		||||
        }).catch(function (err) {
 | 
			
		||||
          return callback(err, null)
 | 
			
		||||
        })
 | 
			
		||||
      },
 | 
			
		||||
      checkAllNotesRevision: function (callback) {
 | 
			
		||||
        Revision.saveAllNotesRevision(function (err, notes) {
 | 
			
		||||
          if (err) return callback(err, null)
 | 
			
		||||
          if (!notes || notes.length <= 0) {
 | 
			
		||||
            return callback(null, notes)
 | 
			
		||||
          } else {
 | 
			
		||||
            Revision.checkAllNotesRevision(callback)
 | 
			
		||||
          }
 | 
			
		||||
        })
 | 
			
		||||
      },
 | 
			
		||||
      saveAllNotesRevision: function (callback) {
 | 
			
		||||
        sequelize.models.Note.findAll({
 | 
			
		||||
          // query all notes that need to save for revision
 | 
			
		||||
          where: {
 | 
			
		||||
            $and: [
 | 
			
		||||
              {
 | 
			
		||||
                lastchangeAt: {
 | 
			
		||||
                  $or: {
 | 
			
		||||
                    $eq: null,
 | 
			
		||||
                    $and: {
 | 
			
		||||
                      $ne: null,
 | 
			
		||||
                      $gt: sequelize.col('createdAt')
 | 
			
		||||
                    }
 | 
			
		||||
                    callback(null, data);
 | 
			
		||||
                }).catch(function (err) {
 | 
			
		||||
                    callback(err, null);
 | 
			
		||||
                });
 | 
			
		||||
            },
 | 
			
		||||
            getPatchedNoteRevisionByTime: function (note, time, callback) {
 | 
			
		||||
                // find all revisions to prepare for all possible calculation
 | 
			
		||||
                Revision.findAll({
 | 
			
		||||
                    where: {
 | 
			
		||||
                        noteId: note.id
 | 
			
		||||
                    },
 | 
			
		||||
                    order: '"createdAt" DESC'
 | 
			
		||||
                }).then(function (revisions) {
 | 
			
		||||
                    if (revisions.length <= 0) return callback(null, null);
 | 
			
		||||
                    // measure target revision position 
 | 
			
		||||
                    Revision.count({
 | 
			
		||||
                        where: {
 | 
			
		||||
                            noteId: note.id,
 | 
			
		||||
                            createdAt: {
 | 
			
		||||
                                $gte: time
 | 
			
		||||
                            }
 | 
			
		||||
                        },
 | 
			
		||||
                        order: '"createdAt" DESC'
 | 
			
		||||
                    }).then(function (count) {
 | 
			
		||||
                        if (count <= 0) return callback(null, null);
 | 
			
		||||
                        sendDmpWorker({
 | 
			
		||||
                            msg: 'get revision',
 | 
			
		||||
                            revisions: revisions,
 | 
			
		||||
                            count: count
 | 
			
		||||
                        }, callback);
 | 
			
		||||
                    }).catch(function (err) {
 | 
			
		||||
                        return callback(err, null);
 | 
			
		||||
                    });
 | 
			
		||||
                }).catch(function (err) {
 | 
			
		||||
                    return callback(err, null);
 | 
			
		||||
                });
 | 
			
		||||
            },
 | 
			
		||||
            checkAllNotesRevision: function (callback) {
 | 
			
		||||
                Revision.saveAllNotesRevision(function (err, notes) {
 | 
			
		||||
                    if (err) return callback(err, null);
 | 
			
		||||
                    if (!notes || notes.length <= 0) {
 | 
			
		||||
                        return callback(null, notes);
 | 
			
		||||
                    } else {
 | 
			
		||||
                        Revision.checkAllNotesRevision(callback);
 | 
			
		||||
                    }
 | 
			
		||||
                });
 | 
			
		||||
            },
 | 
			
		||||
            saveAllNotesRevision: function (callback) {
 | 
			
		||||
                sequelize.models.Note.findAll({
 | 
			
		||||
                    // query all notes that need to save for revision
 | 
			
		||||
                    where: {
 | 
			
		||||
                        $and: [
 | 
			
		||||
                            {
 | 
			
		||||
                                lastchangeAt: {
 | 
			
		||||
                                    $or: {
 | 
			
		||||
                                        $eq: null,
 | 
			
		||||
                                        $and: {
 | 
			
		||||
                                            $ne: null,
 | 
			
		||||
                                            $gt: sequelize.col('createdAt')
 | 
			
		||||
                                        }
 | 
			
		||||
                                    }
 | 
			
		||||
                                }
 | 
			
		||||
                            },
 | 
			
		||||
                            {
 | 
			
		||||
                                savedAt: {
 | 
			
		||||
                                    $or: {
 | 
			
		||||
                                        $eq: null,
 | 
			
		||||
                                        $lt: sequelize.col('lastchangeAt')
 | 
			
		||||
                                    }
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                        ]
 | 
			
		||||
                    }
 | 
			
		||||
                }).then(function (notes) {
 | 
			
		||||
                    if (notes.length <= 0) return callback(null, notes);
 | 
			
		||||
                    var savedNotes = [];
 | 
			
		||||
                    async.each(notes, function (note, _callback) {
 | 
			
		||||
                        // revision saving policy: note not been modified for 5 mins or not save for 10 mins
 | 
			
		||||
                        if (note.lastchangeAt && note.savedAt) {
 | 
			
		||||
                            var lastchangeAt = moment(note.lastchangeAt);
 | 
			
		||||
                            var savedAt = moment(note.savedAt);
 | 
			
		||||
                            if (moment().isAfter(lastchangeAt.add(5, 'minutes'))) {
 | 
			
		||||
                                savedNotes.push(note);
 | 
			
		||||
                                Revision.saveNoteRevision(note, _callback);
 | 
			
		||||
                            } else if (lastchangeAt.isAfter(savedAt.add(10, 'minutes'))) {
 | 
			
		||||
                                savedNotes.push(note);
 | 
			
		||||
                                Revision.saveNoteRevision(note, _callback);
 | 
			
		||||
                            } else {
 | 
			
		||||
                                return _callback(null, null);
 | 
			
		||||
                            }
 | 
			
		||||
                        } else {
 | 
			
		||||
                            savedNotes.push(note);
 | 
			
		||||
                            Revision.saveNoteRevision(note, _callback);
 | 
			
		||||
                        }
 | 
			
		||||
                    }, function (err) {
 | 
			
		||||
                        if (err) return callback(err, null);
 | 
			
		||||
                        // return null when no notes need saving at this moment but have delayed tasks to be done 
 | 
			
		||||
                        var result = ((savedNotes.length == 0) && (notes.length > savedNotes.length)) ? null : savedNotes;
 | 
			
		||||
                        return callback(null, result);
 | 
			
		||||
                    });
 | 
			
		||||
                }).catch(function (err) {
 | 
			
		||||
                    return callback(err, null);
 | 
			
		||||
                });
 | 
			
		||||
            },
 | 
			
		||||
            saveNoteRevision: function (note, callback) {
 | 
			
		||||
                Revision.findAll({
 | 
			
		||||
                    where: {
 | 
			
		||||
                        noteId: note.id
 | 
			
		||||
                    },
 | 
			
		||||
                    order: '"createdAt" DESC'
 | 
			
		||||
                }).then(function (revisions) {
 | 
			
		||||
                    if (revisions.length <= 0) {
 | 
			
		||||
                        // if no revision available
 | 
			
		||||
                        Revision.create({
 | 
			
		||||
                            noteId: note.id,
 | 
			
		||||
                            lastContent: note.content,
 | 
			
		||||
                            length: note.content.length,
 | 
			
		||||
                            authorship: note.authorship
 | 
			
		||||
                        }).then(function (revision) {
 | 
			
		||||
                            Revision.finishSaveNoteRevision(note, revision, callback);
 | 
			
		||||
                        }).catch(function (err) {
 | 
			
		||||
                            return callback(err, null);
 | 
			
		||||
                        });
 | 
			
		||||
                    } else {
 | 
			
		||||
                        var latestRevision = revisions[0];
 | 
			
		||||
                        var lastContent = latestRevision.content || latestRevision.lastContent;
 | 
			
		||||
                        var content = note.content;
 | 
			
		||||
                        sendDmpWorker({
 | 
			
		||||
                            msg: 'create patch',
 | 
			
		||||
                            lastDoc: lastContent,
 | 
			
		||||
                            currDoc: content,
 | 
			
		||||
                        }, function (err, patch) {
 | 
			
		||||
                            if (err) logger.error('save note revision error', err);
 | 
			
		||||
                            if (!patch) {
 | 
			
		||||
                                // if patch is empty (means no difference) then just update the latest revision updated time 
 | 
			
		||||
                                latestRevision.changed('updatedAt', true); 
 | 
			
		||||
                                latestRevision.update({
 | 
			
		||||
                                    updatedAt: Date.now()
 | 
			
		||||
                                }).then(function (revision) {
 | 
			
		||||
                                    Revision.finishSaveNoteRevision(note, revision, callback);
 | 
			
		||||
                                }).catch(function (err) {
 | 
			
		||||
                                    return callback(err, null);
 | 
			
		||||
                                });
 | 
			
		||||
                            } else {
 | 
			
		||||
                                Revision.create({
 | 
			
		||||
                                    noteId: note.id,
 | 
			
		||||
                                    patch: patch,
 | 
			
		||||
                                    content: note.content,
 | 
			
		||||
                                    length: note.content.length,
 | 
			
		||||
                                    authorship: note.authorship
 | 
			
		||||
                                }).then(function (revision) {
 | 
			
		||||
                                    // clear last revision content to reduce db size
 | 
			
		||||
                                    latestRevision.update({
 | 
			
		||||
                                        content: null
 | 
			
		||||
                                    }).then(function () {
 | 
			
		||||
                                        Revision.finishSaveNoteRevision(note, revision, callback);
 | 
			
		||||
                                    }).catch(function (err) {
 | 
			
		||||
                                        return callback(err, null);
 | 
			
		||||
                                    });
 | 
			
		||||
                                }).catch(function (err) {
 | 
			
		||||
                                    return callback(err, null);
 | 
			
		||||
                                });
 | 
			
		||||
                            }
 | 
			
		||||
                        });
 | 
			
		||||
                    }
 | 
			
		||||
                }).catch(function (err) {
 | 
			
		||||
                    return callback(err, null);
 | 
			
		||||
                });
 | 
			
		||||
            },
 | 
			
		||||
            finishSaveNoteRevision: function (note, revision, callback) {
 | 
			
		||||
                note.update({
 | 
			
		||||
                    savedAt: revision.updatedAt
 | 
			
		||||
                }).then(function () {
 | 
			
		||||
                    return callback(null, revision);
 | 
			
		||||
                }).catch(function (err) {
 | 
			
		||||
                    return callback(err, null);
 | 
			
		||||
                });
 | 
			
		||||
                  }
 | 
			
		||||
                }
 | 
			
		||||
              },
 | 
			
		||||
              {
 | 
			
		||||
                savedAt: {
 | 
			
		||||
                  $or: {
 | 
			
		||||
                    $eq: null,
 | 
			
		||||
                    $lt: sequelize.col('lastchangeAt')
 | 
			
		||||
                  }
 | 
			
		||||
                }
 | 
			
		||||
              }
 | 
			
		||||
            ]
 | 
			
		||||
          }
 | 
			
		||||
        }).then(function (notes) {
 | 
			
		||||
          if (notes.length <= 0) return callback(null, notes)
 | 
			
		||||
          var savedNotes = []
 | 
			
		||||
          async.each(notes, function (note, _callback) {
 | 
			
		||||
            // revision saving policy: note not been modified for 5 mins or not save for 10 mins
 | 
			
		||||
            if (note.lastchangeAt && note.savedAt) {
 | 
			
		||||
              var lastchangeAt = moment(note.lastchangeAt)
 | 
			
		||||
              var savedAt = moment(note.savedAt)
 | 
			
		||||
              if (moment().isAfter(lastchangeAt.add(5, 'minutes'))) {
 | 
			
		||||
                savedNotes.push(note)
 | 
			
		||||
                Revision.saveNoteRevision(note, _callback)
 | 
			
		||||
              } else if (lastchangeAt.isAfter(savedAt.add(10, 'minutes'))) {
 | 
			
		||||
                savedNotes.push(note)
 | 
			
		||||
                Revision.saveNoteRevision(note, _callback)
 | 
			
		||||
              } else {
 | 
			
		||||
                return _callback(null, null)
 | 
			
		||||
              }
 | 
			
		||||
            } else {
 | 
			
		||||
              savedNotes.push(note)
 | 
			
		||||
              Revision.saveNoteRevision(note, _callback)
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    });
 | 
			
		||||
          }, function (err) {
 | 
			
		||||
            if (err) {
 | 
			
		||||
              return callback(err, null)
 | 
			
		||||
            }
 | 
			
		||||
            // return null when no notes need saving at this moment but have delayed tasks to be done
 | 
			
		||||
            var result = ((savedNotes.length === 0) && (notes.length > savedNotes.length)) ? null : savedNotes
 | 
			
		||||
            return callback(null, result)
 | 
			
		||||
          })
 | 
			
		||||
        }).catch(function (err) {
 | 
			
		||||
          return callback(err, null)
 | 
			
		||||
        })
 | 
			
		||||
      },
 | 
			
		||||
      saveNoteRevision: function (note, callback) {
 | 
			
		||||
        Revision.findAll({
 | 
			
		||||
          where: {
 | 
			
		||||
            noteId: note.id
 | 
			
		||||
          },
 | 
			
		||||
          order: '"createdAt" DESC'
 | 
			
		||||
        }).then(function (revisions) {
 | 
			
		||||
          if (revisions.length <= 0) {
 | 
			
		||||
            // if no revision available
 | 
			
		||||
            Revision.create({
 | 
			
		||||
              noteId: note.id,
 | 
			
		||||
              lastContent: note.content,
 | 
			
		||||
              length: note.content.length,
 | 
			
		||||
              authorship: note.authorship
 | 
			
		||||
            }).then(function (revision) {
 | 
			
		||||
              Revision.finishSaveNoteRevision(note, revision, callback)
 | 
			
		||||
            }).catch(function (err) {
 | 
			
		||||
              return callback(err, null)
 | 
			
		||||
            })
 | 
			
		||||
          } else {
 | 
			
		||||
            var latestRevision = revisions[0]
 | 
			
		||||
            var lastContent = latestRevision.content || latestRevision.lastContent
 | 
			
		||||
            var content = note.content
 | 
			
		||||
            sendDmpWorker({
 | 
			
		||||
              msg: 'create patch',
 | 
			
		||||
              lastDoc: lastContent,
 | 
			
		||||
              currDoc: content
 | 
			
		||||
            }, function (err, patch) {
 | 
			
		||||
              if (err) logger.error('save note revision error', err)
 | 
			
		||||
              if (!patch) {
 | 
			
		||||
                // if patch is empty (means no difference) then just update the latest revision updated time
 | 
			
		||||
                latestRevision.changed('updatedAt', true)
 | 
			
		||||
                latestRevision.update({
 | 
			
		||||
                  updatedAt: Date.now()
 | 
			
		||||
                }).then(function (revision) {
 | 
			
		||||
                  Revision.finishSaveNoteRevision(note, revision, callback)
 | 
			
		||||
                }).catch(function (err) {
 | 
			
		||||
                  return callback(err, null)
 | 
			
		||||
                })
 | 
			
		||||
              } else {
 | 
			
		||||
                Revision.create({
 | 
			
		||||
                  noteId: note.id,
 | 
			
		||||
                  patch: patch,
 | 
			
		||||
                  content: note.content,
 | 
			
		||||
                  length: note.content.length,
 | 
			
		||||
                  authorship: note.authorship
 | 
			
		||||
                }).then(function (revision) {
 | 
			
		||||
                  // clear last revision content to reduce db size
 | 
			
		||||
                  latestRevision.update({
 | 
			
		||||
                    content: null
 | 
			
		||||
                  }).then(function () {
 | 
			
		||||
                    Revision.finishSaveNoteRevision(note, revision, callback)
 | 
			
		||||
                  }).catch(function (err) {
 | 
			
		||||
                    return callback(err, null)
 | 
			
		||||
                  })
 | 
			
		||||
                }).catch(function (err) {
 | 
			
		||||
                  return callback(err, null)
 | 
			
		||||
                })
 | 
			
		||||
              }
 | 
			
		||||
            })
 | 
			
		||||
          }
 | 
			
		||||
        }).catch(function (err) {
 | 
			
		||||
          return callback(err, null)
 | 
			
		||||
        })
 | 
			
		||||
      },
 | 
			
		||||
      finishSaveNoteRevision: function (note, revision, callback) {
 | 
			
		||||
        note.update({
 | 
			
		||||
          savedAt: revision.updatedAt
 | 
			
		||||
        }).then(function () {
 | 
			
		||||
          return callback(null, revision)
 | 
			
		||||
        }).catch(function (err) {
 | 
			
		||||
          return callback(err, null)
 | 
			
		||||
        })
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
    return Revision;
 | 
			
		||||
};
 | 
			
		||||
  return Revision
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -1,19 +1,17 @@
 | 
			
		||||
"use strict";
 | 
			
		||||
 | 
			
		||||
//external modules
 | 
			
		||||
var shortId = require('shortid');
 | 
			
		||||
// external modules
 | 
			
		||||
var shortId = require('shortid')
 | 
			
		||||
 | 
			
		||||
module.exports = function (sequelize, DataTypes) {
 | 
			
		||||
    var Temp = sequelize.define("Temp", {
 | 
			
		||||
        id: {
 | 
			
		||||
            type: DataTypes.STRING,
 | 
			
		||||
            primaryKey: true,
 | 
			
		||||
            defaultValue: shortId.generate
 | 
			
		||||
        },
 | 
			
		||||
        data: {
 | 
			
		||||
            type: DataTypes.TEXT
 | 
			
		||||
        }
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
    return Temp;
 | 
			
		||||
};
 | 
			
		||||
  var Temp = sequelize.define('Temp', {
 | 
			
		||||
    id: {
 | 
			
		||||
      type: DataTypes.STRING,
 | 
			
		||||
      primaryKey: true,
 | 
			
		||||
      defaultValue: shortId.generate
 | 
			
		||||
    },
 | 
			
		||||
    data: {
 | 
			
		||||
      type: DataTypes.TEXT
 | 
			
		||||
    }
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  return Temp
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -1,149 +1,147 @@
 | 
			
		||||
"use strict";
 | 
			
		||||
 | 
			
		||||
// external modules
 | 
			
		||||
var md5 = require("blueimp-md5");
 | 
			
		||||
var Sequelize = require("sequelize");
 | 
			
		||||
var scrypt = require('scrypt');
 | 
			
		||||
var md5 = require('blueimp-md5')
 | 
			
		||||
var Sequelize = require('sequelize')
 | 
			
		||||
var scrypt = require('scrypt')
 | 
			
		||||
 | 
			
		||||
// core
 | 
			
		||||
var logger = require("../logger.js");
 | 
			
		||||
var letterAvatars = require('../letter-avatars.js');
 | 
			
		||||
var logger = require('../logger.js')
 | 
			
		||||
var letterAvatars = require('../letter-avatars.js')
 | 
			
		||||
 | 
			
		||||
module.exports = function (sequelize, DataTypes) {
 | 
			
		||||
    var User = sequelize.define("User", {
 | 
			
		||||
        id: {
 | 
			
		||||
            type: DataTypes.UUID,
 | 
			
		||||
            primaryKey: true,
 | 
			
		||||
            defaultValue: Sequelize.UUIDV4
 | 
			
		||||
        },
 | 
			
		||||
        profileid: {
 | 
			
		||||
            type: DataTypes.STRING,
 | 
			
		||||
            unique: true
 | 
			
		||||
        },
 | 
			
		||||
        profile: {
 | 
			
		||||
            type: DataTypes.TEXT
 | 
			
		||||
        },
 | 
			
		||||
        history: {
 | 
			
		||||
            type: DataTypes.TEXT
 | 
			
		||||
        },
 | 
			
		||||
        accessToken: {
 | 
			
		||||
            type: DataTypes.STRING
 | 
			
		||||
        },
 | 
			
		||||
        refreshToken: {
 | 
			
		||||
            type: DataTypes.STRING
 | 
			
		||||
        },
 | 
			
		||||
        email: {
 | 
			
		||||
            type: Sequelize.TEXT, 
 | 
			
		||||
            validate: {
 | 
			
		||||
                isEmail: true
 | 
			
		||||
            }
 | 
			
		||||
        },
 | 
			
		||||
        password: {
 | 
			
		||||
            type: Sequelize.TEXT,
 | 
			
		||||
            set: function(value) {
 | 
			
		||||
                var hash = scrypt.kdfSync(value, scrypt.paramsSync(0.1)).toString("hex");
 | 
			
		||||
                this.setDataValue('password', hash);
 | 
			
		||||
            }
 | 
			
		||||
  var User = sequelize.define('User', {
 | 
			
		||||
    id: {
 | 
			
		||||
      type: DataTypes.UUID,
 | 
			
		||||
      primaryKey: true,
 | 
			
		||||
      defaultValue: Sequelize.UUIDV4
 | 
			
		||||
    },
 | 
			
		||||
    profileid: {
 | 
			
		||||
      type: DataTypes.STRING,
 | 
			
		||||
      unique: true
 | 
			
		||||
    },
 | 
			
		||||
    profile: {
 | 
			
		||||
      type: DataTypes.TEXT
 | 
			
		||||
    },
 | 
			
		||||
    history: {
 | 
			
		||||
      type: DataTypes.TEXT
 | 
			
		||||
    },
 | 
			
		||||
    accessToken: {
 | 
			
		||||
      type: DataTypes.STRING
 | 
			
		||||
    },
 | 
			
		||||
    refreshToken: {
 | 
			
		||||
      type: DataTypes.STRING
 | 
			
		||||
    },
 | 
			
		||||
    email: {
 | 
			
		||||
      type: Sequelize.TEXT,
 | 
			
		||||
      validate: {
 | 
			
		||||
        isEmail: true
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    password: {
 | 
			
		||||
      type: Sequelize.TEXT,
 | 
			
		||||
      set: function (value) {
 | 
			
		||||
        var hash = scrypt.kdfSync(value, scrypt.paramsSync(0.1)).toString('hex')
 | 
			
		||||
        this.setDataValue('password', hash)
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }, {
 | 
			
		||||
    instanceMethods: {
 | 
			
		||||
      verifyPassword: function (attempt) {
 | 
			
		||||
        if (scrypt.verifyKdfSync(new Buffer(this.password, 'hex'), attempt)) {
 | 
			
		||||
          return this
 | 
			
		||||
        } else {
 | 
			
		||||
          return false
 | 
			
		||||
        }
 | 
			
		||||
    }, {
 | 
			
		||||
        instanceMethods: {
 | 
			
		||||
            verifyPassword: function(attempt) {
 | 
			
		||||
                if (scrypt.verifyKdfSync(new Buffer(this.password, "hex"), attempt)) {
 | 
			
		||||
                    return this;
 | 
			
		||||
                } else {
 | 
			
		||||
                    return false;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        },
 | 
			
		||||
        classMethods: {
 | 
			
		||||
            associate: function (models) {
 | 
			
		||||
                User.hasMany(models.Note, {
 | 
			
		||||
                    foreignKey: "ownerId",
 | 
			
		||||
                    constraints: false
 | 
			
		||||
                });
 | 
			
		||||
                User.hasMany(models.Note, {
 | 
			
		||||
                    foreignKey: "lastchangeuserId",
 | 
			
		||||
                    constraints: false
 | 
			
		||||
                });
 | 
			
		||||
            },
 | 
			
		||||
            getProfile: function (user) {
 | 
			
		||||
                return user.profile ? User.parseProfile(user.profile) : (user.email ? User.parseProfileByEmail(user.email) : null);
 | 
			
		||||
            },
 | 
			
		||||
            parseProfile: function (profile) {
 | 
			
		||||
                try {
 | 
			
		||||
                    var profile = JSON.parse(profile);
 | 
			
		||||
                } catch (err) {
 | 
			
		||||
                    logger.error(err);
 | 
			
		||||
                    profile = null;
 | 
			
		||||
                }
 | 
			
		||||
                if (profile) {
 | 
			
		||||
                    profile = {
 | 
			
		||||
                        name: profile.displayName || profile.username,
 | 
			
		||||
                        photo: User.parsePhotoByProfile(profile),
 | 
			
		||||
                        biggerphoto: User.parsePhotoByProfile(profile, true)
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                return profile;
 | 
			
		||||
            },
 | 
			
		||||
            parsePhotoByProfile: function (profile, bigger) {
 | 
			
		||||
                var photo = null;
 | 
			
		||||
                switch (profile.provider) {
 | 
			
		||||
                    case "facebook":
 | 
			
		||||
                        photo = 'https://graph.facebook.com/' + profile.id + '/picture';
 | 
			
		||||
                        if (bigger) photo += '?width=400';
 | 
			
		||||
                        else photo += '?width=96';
 | 
			
		||||
                        break;
 | 
			
		||||
                    case "twitter":
 | 
			
		||||
                        photo = 'https://twitter.com/' + profile.username + '/profile_image';
 | 
			
		||||
                        if (bigger) photo += '?size=original';
 | 
			
		||||
                        else photo += '?size=bigger';
 | 
			
		||||
                        break;
 | 
			
		||||
                    case "github":
 | 
			
		||||
                        photo = 'https://avatars.githubusercontent.com/u/' + profile.id;
 | 
			
		||||
                        if (bigger) photo += '?s=400';
 | 
			
		||||
                        else photo += '?s=96';
 | 
			
		||||
                        break;
 | 
			
		||||
                    case "gitlab":
 | 
			
		||||
                        photo = profile.avatarUrl;
 | 
			
		||||
                        if (bigger) photo = photo.replace(/(\?s=)\d*$/i, '$1400');
 | 
			
		||||
                        else photo = photo.replace(/(\?s=)\d*$/i, '$196');
 | 
			
		||||
                        break;
 | 
			
		||||
                    case "dropbox":
 | 
			
		||||
                        //no image api provided, use gravatar
 | 
			
		||||
                        photo = 'https://www.gravatar.com/avatar/' + md5(profile.emails[0].value);
 | 
			
		||||
                        if (bigger) photo += '?s=400';
 | 
			
		||||
                        else photo += '?s=96';
 | 
			
		||||
                        break;
 | 
			
		||||
                    case "google":
 | 
			
		||||
                        photo = profile.photos[0].value;
 | 
			
		||||
                        if (bigger) photo = photo.replace(/(\?sz=)\d*$/i, '$1400');
 | 
			
		||||
                        else photo = photo.replace(/(\?sz=)\d*$/i, '$196');
 | 
			
		||||
                        break;
 | 
			
		||||
                    case "ldap":
 | 
			
		||||
                        //no image api provided,
 | 
			
		||||
                        //use gravatar if email exists,
 | 
			
		||||
                        //otherwise generate a letter avatar
 | 
			
		||||
                        if (profile.emails[0]) {
 | 
			
		||||
                            photo = 'https://www.gravatar.com/avatar/' + md5(profile.emails[0]);
 | 
			
		||||
                            if (bigger) photo += '?s=400';
 | 
			
		||||
                            else photo += '?s=96';
 | 
			
		||||
                        } else {
 | 
			
		||||
                            photo = letterAvatars(profile.username);
 | 
			
		||||
                        }
 | 
			
		||||
                        break;
 | 
			
		||||
                }
 | 
			
		||||
                return photo;
 | 
			
		||||
            },
 | 
			
		||||
            parseProfileByEmail: function (email) {
 | 
			
		||||
                var photoUrl = 'https://www.gravatar.com/avatar/' + md5(email);
 | 
			
		||||
                return {
 | 
			
		||||
                    name: email.substring(0, email.lastIndexOf("@")),
 | 
			
		||||
                    photo: photoUrl += '?s=96',
 | 
			
		||||
                    biggerphoto: photoUrl += '?s=400'
 | 
			
		||||
                };
 | 
			
		||||
            }
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    classMethods: {
 | 
			
		||||
      associate: function (models) {
 | 
			
		||||
        User.hasMany(models.Note, {
 | 
			
		||||
          foreignKey: 'ownerId',
 | 
			
		||||
          constraints: false
 | 
			
		||||
        })
 | 
			
		||||
        User.hasMany(models.Note, {
 | 
			
		||||
          foreignKey: 'lastchangeuserId',
 | 
			
		||||
          constraints: false
 | 
			
		||||
        })
 | 
			
		||||
      },
 | 
			
		||||
      getProfile: function (user) {
 | 
			
		||||
        return user.profile ? User.parseProfile(user.profile) : (user.email ? User.parseProfileByEmail(user.email) : null)
 | 
			
		||||
      },
 | 
			
		||||
      parseProfile: function (profile) {
 | 
			
		||||
        try {
 | 
			
		||||
          profile = JSON.parse(profile)
 | 
			
		||||
        } catch (err) {
 | 
			
		||||
          logger.error(err)
 | 
			
		||||
          profile = null
 | 
			
		||||
        }
 | 
			
		||||
    });
 | 
			
		||||
        if (profile) {
 | 
			
		||||
          profile = {
 | 
			
		||||
            name: profile.displayName || profile.username,
 | 
			
		||||
            photo: User.parsePhotoByProfile(profile),
 | 
			
		||||
            biggerphoto: User.parsePhotoByProfile(profile, true)
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
        return profile
 | 
			
		||||
      },
 | 
			
		||||
      parsePhotoByProfile: function (profile, bigger) {
 | 
			
		||||
        var photo = null
 | 
			
		||||
        switch (profile.provider) {
 | 
			
		||||
          case 'facebook':
 | 
			
		||||
            photo = 'https://graph.facebook.com/' + profile.id + '/picture'
 | 
			
		||||
            if (bigger) photo += '?width=400'
 | 
			
		||||
            else photo += '?width=96'
 | 
			
		||||
            break
 | 
			
		||||
          case 'twitter':
 | 
			
		||||
            photo = 'https://twitter.com/' + profile.username + '/profile_image'
 | 
			
		||||
            if (bigger) photo += '?size=original'
 | 
			
		||||
            else photo += '?size=bigger'
 | 
			
		||||
            break
 | 
			
		||||
          case 'github':
 | 
			
		||||
            photo = 'https://avatars.githubusercontent.com/u/' + profile.id
 | 
			
		||||
            if (bigger) photo += '?s=400'
 | 
			
		||||
            else photo += '?s=96'
 | 
			
		||||
            break
 | 
			
		||||
          case 'gitlab':
 | 
			
		||||
            photo = profile.avatarUrl
 | 
			
		||||
            if (bigger) photo = photo.replace(/(\?s=)\d*$/i, '$1400')
 | 
			
		||||
            else photo = photo.replace(/(\?s=)\d*$/i, '$196')
 | 
			
		||||
            break
 | 
			
		||||
          case 'dropbox':
 | 
			
		||||
            // no image api provided, use gravatar
 | 
			
		||||
            photo = 'https://www.gravatar.com/avatar/' + md5(profile.emails[0].value)
 | 
			
		||||
            if (bigger) photo += '?s=400'
 | 
			
		||||
            else photo += '?s=96'
 | 
			
		||||
            break
 | 
			
		||||
          case 'google':
 | 
			
		||||
            photo = profile.photos[0].value
 | 
			
		||||
            if (bigger) photo = photo.replace(/(\?sz=)\d*$/i, '$1400')
 | 
			
		||||
            else photo = photo.replace(/(\?sz=)\d*$/i, '$196')
 | 
			
		||||
            break
 | 
			
		||||
          case 'ldap':
 | 
			
		||||
            // no image api provided,
 | 
			
		||||
            // use gravatar if email exists,
 | 
			
		||||
            // otherwise generate a letter avatar
 | 
			
		||||
            if (profile.emails[0]) {
 | 
			
		||||
              photo = 'https://www.gravatar.com/avatar/' + md5(profile.emails[0])
 | 
			
		||||
              if (bigger) photo += '?s=400'
 | 
			
		||||
              else photo += '?s=96'
 | 
			
		||||
            } else {
 | 
			
		||||
              photo = letterAvatars(profile.username)
 | 
			
		||||
            }
 | 
			
		||||
            break
 | 
			
		||||
        }
 | 
			
		||||
        return photo
 | 
			
		||||
      },
 | 
			
		||||
      parseProfileByEmail: function (email) {
 | 
			
		||||
        var photoUrl = 'https://www.gravatar.com/avatar/' + md5(email)
 | 
			
		||||
        return {
 | 
			
		||||
          name: email.substring(0, email.lastIndexOf('@')),
 | 
			
		||||
          photo: photoUrl + '?s=96',
 | 
			
		||||
          biggerphoto: photoUrl + '?s=400'
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
    return User;
 | 
			
		||||
};
 | 
			
		||||
  return User
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										1793
									
								
								lib/realtime.js
									
									
									
									
									
								
							
							
						
						
									
										1793
									
								
								lib/realtime.js
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1128
									
								
								lib/response.js
									
									
									
									
									
								
							
							
						
						
									
										1128
									
								
								lib/response.js
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							@ -1,140 +1,137 @@
 | 
			
		||||
// external modules
 | 
			
		||||
var DiffMatchPatch = require('diff-match-patch');
 | 
			
		||||
var dmp = new DiffMatchPatch();
 | 
			
		||||
var DiffMatchPatch = require('diff-match-patch')
 | 
			
		||||
var dmp = new DiffMatchPatch()
 | 
			
		||||
 | 
			
		||||
// core
 | 
			
		||||
var config = require("../config.js");
 | 
			
		||||
var logger = require("../logger.js");
 | 
			
		||||
var config = require('../config.js')
 | 
			
		||||
var logger = require('../logger.js')
 | 
			
		||||
 | 
			
		||||
process.on('message', function(data) {
 | 
			
		||||
    if (!data || !data.msg || !data.cacheKey) {
 | 
			
		||||
        return logger.error('dmp worker error: not enough data');
 | 
			
		||||
    }
 | 
			
		||||
    switch (data.msg) {
 | 
			
		||||
        case 'create patch':
 | 
			
		||||
            if (!data.hasOwnProperty('lastDoc') || !data.hasOwnProperty('currDoc')) {
 | 
			
		||||
                return logger.error('dmp worker error: not enough data on create patch');
 | 
			
		||||
            }
 | 
			
		||||
            try {
 | 
			
		||||
                var patch = createPatch(data.lastDoc, data.currDoc);
 | 
			
		||||
                process.send({
 | 
			
		||||
                    msg: 'check',
 | 
			
		||||
                    result: patch,
 | 
			
		||||
                    cacheKey: data.cacheKey
 | 
			
		||||
                });
 | 
			
		||||
            } catch (err) {
 | 
			
		||||
                logger.error('dmp worker error', err);
 | 
			
		||||
                process.send({
 | 
			
		||||
                    msg: 'error',
 | 
			
		||||
                    error: err,
 | 
			
		||||
                    cacheKey: data.cacheKey
 | 
			
		||||
                });
 | 
			
		||||
            }
 | 
			
		||||
            break;
 | 
			
		||||
        case 'get revision':
 | 
			
		||||
            if (!data.hasOwnProperty('revisions') || !data.hasOwnProperty('count')) {
 | 
			
		||||
                return logger.error('dmp worker error: not enough data on get revision');
 | 
			
		||||
            }
 | 
			
		||||
            try {
 | 
			
		||||
                var result = getRevision(data.revisions, data.count);
 | 
			
		||||
                process.send({
 | 
			
		||||
                    msg: 'check',
 | 
			
		||||
                    result: result,
 | 
			
		||||
                    cacheKey: data.cacheKey
 | 
			
		||||
                });
 | 
			
		||||
            } catch (err) {
 | 
			
		||||
                logger.error('dmp worker error', err);
 | 
			
		||||
                process.send({
 | 
			
		||||
                    msg: 'error',
 | 
			
		||||
                    error: err,
 | 
			
		||||
                    cacheKey: data.cacheKey
 | 
			
		||||
                });
 | 
			
		||||
            }
 | 
			
		||||
            break;
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
process.on('message', function (data) {
 | 
			
		||||
  if (!data || !data.msg || !data.cacheKey) {
 | 
			
		||||
    return logger.error('dmp worker error: not enough data')
 | 
			
		||||
  }
 | 
			
		||||
  switch (data.msg) {
 | 
			
		||||
    case 'create patch':
 | 
			
		||||
      if (!data.hasOwnProperty('lastDoc') || !data.hasOwnProperty('currDoc')) {
 | 
			
		||||
        return logger.error('dmp worker error: not enough data on create patch')
 | 
			
		||||
      }
 | 
			
		||||
      try {
 | 
			
		||||
        var patch = createPatch(data.lastDoc, data.currDoc)
 | 
			
		||||
        process.send({
 | 
			
		||||
          msg: 'check',
 | 
			
		||||
          result: patch,
 | 
			
		||||
          cacheKey: data.cacheKey
 | 
			
		||||
        })
 | 
			
		||||
      } catch (err) {
 | 
			
		||||
        logger.error('dmp worker error', err)
 | 
			
		||||
        process.send({
 | 
			
		||||
          msg: 'error',
 | 
			
		||||
          error: err,
 | 
			
		||||
          cacheKey: data.cacheKey
 | 
			
		||||
        })
 | 
			
		||||
      }
 | 
			
		||||
      break
 | 
			
		||||
    case 'get revision':
 | 
			
		||||
      if (!data.hasOwnProperty('revisions') || !data.hasOwnProperty('count')) {
 | 
			
		||||
        return logger.error('dmp worker error: not enough data on get revision')
 | 
			
		||||
      }
 | 
			
		||||
      try {
 | 
			
		||||
        var result = getRevision(data.revisions, data.count)
 | 
			
		||||
        process.send({
 | 
			
		||||
          msg: 'check',
 | 
			
		||||
          result: result,
 | 
			
		||||
          cacheKey: data.cacheKey
 | 
			
		||||
        })
 | 
			
		||||
      } catch (err) {
 | 
			
		||||
        logger.error('dmp worker error', err)
 | 
			
		||||
        process.send({
 | 
			
		||||
          msg: 'error',
 | 
			
		||||
          error: err,
 | 
			
		||||
          cacheKey: data.cacheKey
 | 
			
		||||
        })
 | 
			
		||||
      }
 | 
			
		||||
      break
 | 
			
		||||
  }
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
function createPatch(lastDoc, currDoc) {
 | 
			
		||||
    var ms_start = (new Date()).getTime();
 | 
			
		||||
    var diff = dmp.diff_main(lastDoc, currDoc);
 | 
			
		||||
    var patch = dmp.patch_make(lastDoc, diff);
 | 
			
		||||
    patch = dmp.patch_toText(patch);
 | 
			
		||||
    var ms_end = (new Date()).getTime();
 | 
			
		||||
    if (config.debug) {
 | 
			
		||||
        logger.info(patch);
 | 
			
		||||
        logger.info((ms_end - ms_start) + 'ms');
 | 
			
		||||
    }
 | 
			
		||||
    return patch;
 | 
			
		||||
function createPatch (lastDoc, currDoc) {
 | 
			
		||||
  var msStart = (new Date()).getTime()
 | 
			
		||||
  var diff = dmp.diff_main(lastDoc, currDoc)
 | 
			
		||||
  var patch = dmp.patch_make(lastDoc, diff)
 | 
			
		||||
  patch = dmp.patch_toText(patch)
 | 
			
		||||
  var msEnd = (new Date()).getTime()
 | 
			
		||||
  if (config.debug) {
 | 
			
		||||
    logger.info(patch)
 | 
			
		||||
    logger.info((msEnd - msStart) + 'ms')
 | 
			
		||||
  }
 | 
			
		||||
  return patch
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function getRevision(revisions, count) {
 | 
			
		||||
    var ms_start = (new Date()).getTime();
 | 
			
		||||
    var startContent = null;
 | 
			
		||||
    var lastPatch = [];
 | 
			
		||||
    var applyPatches = [];
 | 
			
		||||
    var authorship = [];
 | 
			
		||||
    if (count <= Math.round(revisions.length / 2)) {
 | 
			
		||||
        // start from top to target
 | 
			
		||||
        for (var i = 0; i < count; i++) {
 | 
			
		||||
            var revision = revisions[i];
 | 
			
		||||
            if (i == 0) {
 | 
			
		||||
                startContent = revision.content || revision.lastContent;
 | 
			
		||||
            }
 | 
			
		||||
            if (i != count - 1) {
 | 
			
		||||
                var patch = dmp.patch_fromText(revision.patch);
 | 
			
		||||
                applyPatches = applyPatches.concat(patch);
 | 
			
		||||
            }
 | 
			
		||||
            lastPatch = revision.patch;
 | 
			
		||||
            authorship = revision.authorship;
 | 
			
		||||
        }
 | 
			
		||||
        // swap DIFF_INSERT and DIFF_DELETE to achieve unpatching
 | 
			
		||||
        for (var i = 0, l = applyPatches.length; i < l; i++) {
 | 
			
		||||
            for (var j = 0, m = applyPatches[i].diffs.length; j < m; j++) {
 | 
			
		||||
                var diff = applyPatches[i].diffs[j];
 | 
			
		||||
                if (diff[0] == DiffMatchPatch.DIFF_INSERT)
 | 
			
		||||
                    diff[0] = DiffMatchPatch.DIFF_DELETE;
 | 
			
		||||
                else if (diff[0] == DiffMatchPatch.DIFF_DELETE)
 | 
			
		||||
                    diff[0] = DiffMatchPatch.DIFF_INSERT;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    } else {
 | 
			
		||||
        // start from bottom to target
 | 
			
		||||
        var l = revisions.length - 1;
 | 
			
		||||
        for (var i = l; i >= count - 1; i--) {
 | 
			
		||||
            var revision = revisions[i];
 | 
			
		||||
            if (i == l) {
 | 
			
		||||
                startContent = revision.lastContent;
 | 
			
		||||
                authorship = revision.authorship;
 | 
			
		||||
            }
 | 
			
		||||
            if (revision.patch) {
 | 
			
		||||
                var patch = dmp.patch_fromText(revision.patch);
 | 
			
		||||
                applyPatches = applyPatches.concat(patch);
 | 
			
		||||
            }
 | 
			
		||||
            lastPatch = revision.patch;
 | 
			
		||||
            authorship = revision.authorship;
 | 
			
		||||
        }
 | 
			
		||||
function getRevision (revisions, count) {
 | 
			
		||||
  var msStart = (new Date()).getTime()
 | 
			
		||||
  var startContent = null
 | 
			
		||||
  var lastPatch = []
 | 
			
		||||
  var applyPatches = []
 | 
			
		||||
  var authorship = []
 | 
			
		||||
  if (count <= Math.round(revisions.length / 2)) {
 | 
			
		||||
    // start from top to target
 | 
			
		||||
    for (let i = 0; i < count; i++) {
 | 
			
		||||
      let revision = revisions[i]
 | 
			
		||||
      if (i === 0) {
 | 
			
		||||
        startContent = revision.content || revision.lastContent
 | 
			
		||||
      }
 | 
			
		||||
      if (i !== count - 1) {
 | 
			
		||||
        let patch = dmp.patch_fromText(revision.patch)
 | 
			
		||||
        applyPatches = applyPatches.concat(patch)
 | 
			
		||||
      }
 | 
			
		||||
      lastPatch = revision.patch
 | 
			
		||||
      authorship = revision.authorship
 | 
			
		||||
    }
 | 
			
		||||
    try {
 | 
			
		||||
        var finalContent = dmp.patch_apply(applyPatches, startContent)[0];
 | 
			
		||||
    } catch (err) {
 | 
			
		||||
        throw new Error(err);
 | 
			
		||||
    // swap DIFF_INSERT and DIFF_DELETE to achieve unpatching
 | 
			
		||||
    for (let i = 0, l = applyPatches.length; i < l; i++) {
 | 
			
		||||
      for (let j = 0, m = applyPatches[i].diffs.length; j < m; j++) {
 | 
			
		||||
        var diff = applyPatches[i].diffs[j]
 | 
			
		||||
        if (diff[0] === DiffMatchPatch.DIFF_INSERT) { diff[0] = DiffMatchPatch.DIFF_DELETE } else if (diff[0] === DiffMatchPatch.DIFF_DELETE) { diff[0] = DiffMatchPatch.DIFF_INSERT }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    var data = {
 | 
			
		||||
        content: finalContent,
 | 
			
		||||
        patch: dmp.patch_fromText(lastPatch),
 | 
			
		||||
        authorship: authorship
 | 
			
		||||
    };
 | 
			
		||||
    var ms_end = (new Date()).getTime();
 | 
			
		||||
    if (config.debug) {
 | 
			
		||||
        logger.info((ms_end - ms_start) + 'ms');
 | 
			
		||||
  } else {
 | 
			
		||||
    // start from bottom to target
 | 
			
		||||
    var l = revisions.length - 1
 | 
			
		||||
    for (var i = l; i >= count - 1; i--) {
 | 
			
		||||
      let revision = revisions[i]
 | 
			
		||||
      if (i === l) {
 | 
			
		||||
        startContent = revision.lastContent
 | 
			
		||||
        authorship = revision.authorship
 | 
			
		||||
      }
 | 
			
		||||
      if (revision.patch) {
 | 
			
		||||
        let patch = dmp.patch_fromText(revision.patch)
 | 
			
		||||
        applyPatches = applyPatches.concat(patch)
 | 
			
		||||
      }
 | 
			
		||||
      lastPatch = revision.patch
 | 
			
		||||
      authorship = revision.authorship
 | 
			
		||||
    }
 | 
			
		||||
    return data;
 | 
			
		||||
  }
 | 
			
		||||
  try {
 | 
			
		||||
    var finalContent = dmp.patch_apply(applyPatches, startContent)[0]
 | 
			
		||||
  } catch (err) {
 | 
			
		||||
    throw new Error(err)
 | 
			
		||||
  }
 | 
			
		||||
  var data = {
 | 
			
		||||
    content: finalContent,
 | 
			
		||||
    patch: dmp.patch_fromText(lastPatch),
 | 
			
		||||
    authorship: authorship
 | 
			
		||||
  }
 | 
			
		||||
  var msEnd = (new Date()).getTime()
 | 
			
		||||
  if (config.debug) {
 | 
			
		||||
    logger.info((msEnd - msStart) + 'ms')
 | 
			
		||||
  }
 | 
			
		||||
  return data
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// log uncaught exception
 | 
			
		||||
process.on('uncaughtException', function (err) {
 | 
			
		||||
    logger.error('An uncaught exception has occured.');
 | 
			
		||||
    logger.error(err);
 | 
			
		||||
    logger.error('Process will exit now.');
 | 
			
		||||
    process.exit(1);
 | 
			
		||||
});
 | 
			
		||||
  logger.error('An uncaught exception has occured.')
 | 
			
		||||
  logger.error(err)
 | 
			
		||||
  logger.error('Process will exit now.')
 | 
			
		||||
  process.exit(1)
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
@ -5,7 +5,7 @@
 | 
			
		||||
  "main": "app.js",
 | 
			
		||||
  "license": "MIT",
 | 
			
		||||
  "scripts": {
 | 
			
		||||
    "test": "npm run-script lint",
 | 
			
		||||
    "test": "node ./node_modules/standard/bin/cmd.js && npm run-script lint",
 | 
			
		||||
    "lint": "eslint .",
 | 
			
		||||
    "dev": "webpack --config webpack.config.js --progress --colors --watch",
 | 
			
		||||
    "build": "webpack --config webpack.production.js --progress --colors",
 | 
			
		||||
@ -165,8 +165,15 @@
 | 
			
		||||
    "optimize-css-assets-webpack-plugin": "^1.3.0",
 | 
			
		||||
    "script-loader": "^0.7.0",
 | 
			
		||||
    "style-loader": "^0.13.1",
 | 
			
		||||
    "standard": "^9.0.1",
 | 
			
		||||
    "url-loader": "^0.5.7",
 | 
			
		||||
    "webpack": "^1.14.0",
 | 
			
		||||
    "webpack-parallel-uglify-plugin": "^0.2.0"
 | 
			
		||||
  },
 | 
			
		||||
  "standard": {
 | 
			
		||||
    "ignore": [
 | 
			
		||||
      "lib/ot",
 | 
			
		||||
      "public/vendor"
 | 
			
		||||
    ]
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user