Files
kami-parse-server/src/index.js

260 lines
8.7 KiB
JavaScript
Raw Normal View History

2016-01-28 10:58:12 -08:00
// ParseServer - open-source compatible API Server for Parse apps
2016-02-20 23:45:54 -08:00
import 'babel-polyfill';
2016-01-28 10:58:12 -08:00
var batch = require('./batch'),
bodyParser = require('body-parser'),
cache = require('./cache'),
DatabaseAdapter = require('./DatabaseAdapter'),
express = require('express'),
middlewares = require('./middlewares'),
multer = require('multer'),
Parse = require('parse/node').Parse,
httpRequest = require('./httpRequest');
2016-02-23 12:54:20 -08:00
2016-02-19 23:47:44 -05:00
import PromiseRouter from './PromiseRouter';
import { GridStoreAdapter } from './Adapters/Files/GridStoreAdapter';
import { S3Adapter } from './Adapters/Files/S3Adapter';
import { FilesController } from './Controllers/FilesController';
2016-02-08 22:51:58 -08:00
import ParsePushAdapter from './Adapters/Push/ParsePushAdapter';
import { PushController } from './Controllers/PushController';
import { ClassesRouter } from './Routers/ClassesRouter';
import { InstallationsRouter } from './Routers/InstallationsRouter';
import { UsersRouter } from './Routers/UsersRouter';
import { SessionsRouter } from './Routers/SessionsRouter';
import { RolesRouter } from './Routers/RolesRouter';
2016-02-19 23:47:44 -05:00
import { AnalyticsRouter } from './Routers/AnalyticsRouter';
import { FunctionsRouter } from './Routers/FunctionsRouter';
import { SchemasRouter } from './Routers/SchemasRouter';
import { IAPValidationRouter } from './Routers/IAPValidationRouter';
import { PushRouter } from './Routers/PushRouter';
import { FilesRouter } from './Routers/FilesRouter';
2016-02-23 12:54:20 -08:00
import { LogsRouter } from './Routers/LogsRouter';
2016-02-23 12:54:20 -08:00
import { loadAdapter } from './Adapters/AdapterLoader';
import { FileLoggerAdapter } from './Adapters/Logger/FileLoggerAdapter';
import { LoggerController } from './Controllers/LoggerController';
2016-02-23 12:54:20 -08:00
import requiredParameter from './requiredParameter';
2016-01-28 10:58:12 -08:00
// Mutate the Parse object to add the Cloud Code handlers
addParseCloud();
// ParseServer works like a constructor of an express app.
// The args that we understand are:
// "databaseAdapter": a class like ExportAdapter providing create, find,
// update, and delete
// "filesAdapter": a class like GridStoreAdapter providing create, get,
// and delete
// "databaseURI": a uri like mongodb://localhost:27017/dbname to tell us
// what database this Parse API connects to.
// "cloud": relative location to cloud code to require, or a function
// that is given an instance of Parse as a parameter. Use this instance of Parse
// to register your cloud code hooks and functions.
2016-01-28 10:58:12 -08:00
// "appId": the application id to host
// "masterKey": the master key for requests to this app
2016-01-30 15:20:44 -08:00
// "facebookAppIds": an array of valid Facebook Application IDs, required
// if using Facebook login
2016-01-28 10:58:12 -08:00
// "collectionPrefix": optional prefix for database collection names
// "fileKey": optional key from Parse dashboard for supporting older files
// hosted by Parse
// "clientKey": optional key from Parse dashboard
// "dotNetKey": optional key from Parse dashboard
// "restAPIKey": optional key from Parse dashboard
// "javascriptKey": optional key from Parse dashboard
// "push": optional key from configure push
function ParseServer({
2016-02-23 12:54:20 -08:00
appId = requiredParameter('You must provide an appId!'),
masterKey = requiredParameter('You must provide a masterKey!'),
databaseAdapter,
filesAdapter,
push,
loggerAdapter,
databaseURI,
cloud,
collectionPrefix = '',
clientKey = '',
javascriptKey = '',
dotNetKey = '',
restAPIKey = '',
fileKey = 'invalid-file-key',
facebookAppIds = [],
enableAnonymousUsers = true,
oauth = {},
2016-02-23 12:54:20 -08:00
serverURL = requiredParameter('You must provide a serverURL!'),
2016-02-23 11:49:21 -05:00
maxUploadSize = '20mb'
}) {
if (databaseAdapter) {
DatabaseAdapter.setAdapter(databaseAdapter);
2016-01-28 10:58:12 -08:00
}
if (databaseURI) {
DatabaseAdapter.setAppDatabaseURI(appId, databaseURI);
2016-01-28 10:58:12 -08:00
}
if (cloud) {
2016-01-28 10:58:12 -08:00
addParseCloud();
if (typeof cloud === 'function') {
cloud(Parse)
} else if (typeof cloud === 'string') {
require(cloud);
2016-02-01 15:41:22 -05:00
} else {
throw "argument 'cloud' must either be a string or a function";
}
2016-01-28 10:58:12 -08:00
}
2016-02-23 12:54:20 -08:00
const filesControllerAdapter = loadAdapter(filesAdapter, GridStoreAdapter);
const pushControllerAdapter = loadAdapter(push, ParsePushAdapter);
const loggerControllerAdapter = loadAdapter(loggerAdapter, FileLoggerAdapter);
2016-01-28 10:58:12 -08:00
// We pass the options and the base class for the adatper,
// Note that passing an instance would work too
const filesController = new FilesController(filesControllerAdapter);
const pushController = new PushController(pushControllerAdapter);
const loggerController = new LoggerController(loggerControllerAdapter);
2016-02-23 12:54:20 -08:00
cache.apps[appId] = {
masterKey: masterKey,
collectionPrefix: collectionPrefix,
clientKey: clientKey,
javascriptKey: javascriptKey,
dotNetKey: dotNetKey,
restAPIKey: restAPIKey,
fileKey: fileKey,
facebookAppIds: facebookAppIds,
filesController: filesController,
pushController: pushController,
loggerController: loggerController,
enableAnonymousUsers: enableAnonymousUsers,
oauth: oauth,
};
2016-01-28 10:58:12 -08:00
2016-01-30 15:20:44 -08:00
// To maintain compatibility. TODO: Remove in v2.1
if (process.env.FACEBOOK_APP_ID) {
cache.apps[appId]['facebookAppIds'].push(process.env.FACEBOOK_APP_ID);
2016-01-30 15:20:44 -08:00
}
2016-01-28 10:58:12 -08:00
// Initialize the node client SDK automatically
Parse.initialize(appId, javascriptKey, masterKey);
Parse.serverURL = serverURL;
2016-01-28 10:58:12 -08:00
// This app serves the Parse API directly.
// It's the equivalent of https://api.parse.com/1 in the hosted Parse API.
var api = express();
// File handling needs to be before default middlewares are applied
2016-02-23 11:49:21 -05:00
api.use('/', new FilesRouter().getExpressRouter({
maxUploadSize: maxUploadSize
}));
2016-01-28 10:58:12 -08:00
// TODO: separate this from the regular ParseServer object
if (process.env.TESTING == 1) {
api.use('/', require('./testing-routes').router);
}
2016-02-23 11:49:21 -05:00
api.use(bodyParser.json({ 'type': '*/*' , limit: maxUploadSize }));
2016-01-28 10:58:12 -08:00
api.use(middlewares.allowCrossDomain);
api.use(middlewares.allowMethodOverride);
api.use(middlewares.handleParseHeaders);
2016-02-09 20:45:02 -08:00
let routers = [
2016-02-19 23:47:44 -05:00
new ClassesRouter(),
new UsersRouter(),
new SessionsRouter(),
new RolesRouter(),
new AnalyticsRouter(),
new InstallationsRouter(),
new FunctionsRouter(),
new SchemasRouter(),
new PushRouter(),
new LogsRouter(),
2016-02-19 23:47:44 -05:00
new IAPValidationRouter()
2016-02-09 20:45:02 -08:00
];
2016-02-23 12:54:20 -08:00
if (process.env.PARSE_EXPERIMENTAL_CONFIG_ENABLED || process.env.TESTING) {
2016-02-09 20:45:02 -08:00
routers.push(require('./global_config'));
}
2016-01-28 10:58:12 -08:00
2016-02-09 20:45:02 -08:00
let appRouter = new PromiseRouter();
routers.forEach((router) => {
appRouter.merge(router);
});
batch.mountOnto(appRouter);
2016-01-28 10:58:12 -08:00
2016-02-09 20:45:02 -08:00
appRouter.mountOnto(api);
2016-01-28 10:58:12 -08:00
api.use(middlewares.handleParseErrors);
process.on('uncaughtException', (err) => {
if( err.code === "EADDRINUSE" ) { // user-friendly message for this common error
console.log(`Unable to listen on port ${err.port}. The port is already in use.`);
process.exit(0);
}
else {
throw err;
}
});
2016-01-28 10:58:12 -08:00
return api;
}
function addParseCloud() {
Parse.Cloud.Functions = {};
Parse.Cloud.Validators = {};
2016-01-28 10:58:12 -08:00
Parse.Cloud.Triggers = {
beforeSave: {},
beforeDelete: {},
afterSave: {},
afterDelete: {}
};
2016-02-03 21:51:40 +00:00
2016-02-24 01:23:24 -08:00
function validateClassNameForTriggers(className) {
const restrictedClassNames = [ '_Session' ];
if (restrictedClassNames.indexOf(className) != -1) {
throw `Triggers are not supported for ${className} class.`;
}
}
Parse.Cloud.define = function(functionName, handler, validationHandler) {
2016-01-28 10:58:12 -08:00
Parse.Cloud.Functions[functionName] = handler;
Parse.Cloud.Validators[functionName] = validationHandler;
2016-01-28 10:58:12 -08:00
};
Parse.Cloud.beforeSave = function(parseClass, handler) {
2016-02-24 01:23:24 -08:00
let className = getClassName(parseClass);
validateClassNameForTriggers(className);
2016-01-28 10:58:12 -08:00
Parse.Cloud.Triggers.beforeSave[className] = handler;
};
Parse.Cloud.beforeDelete = function(parseClass, handler) {
2016-02-24 01:23:24 -08:00
let className = getClassName(parseClass);
validateClassNameForTriggers(className);
2016-01-28 10:58:12 -08:00
Parse.Cloud.Triggers.beforeDelete[className] = handler;
};
Parse.Cloud.afterSave = function(parseClass, handler) {
2016-02-24 01:23:24 -08:00
let className = getClassName(parseClass);
validateClassNameForTriggers(className);
2016-01-28 10:58:12 -08:00
Parse.Cloud.Triggers.afterSave[className] = handler;
};
Parse.Cloud.afterDelete = function(parseClass, handler) {
2016-02-24 01:23:24 -08:00
let className = getClassName(parseClass);
validateClassNameForTriggers(className);
2016-01-28 10:58:12 -08:00
Parse.Cloud.Triggers.afterDelete[className] = handler;
};
Parse.Cloud.httpRequest = httpRequest;
2016-01-28 10:58:12 -08:00
global.Parse = Parse;
}
function getClassName(parseClass) {
if (parseClass && parseClass.className) {
return parseClass.className;
}
return parseClass;
}
module.exports = {
ParseServer: ParseServer,
S3Adapter: S3Adapter
2016-01-28 10:58:12 -08:00
};