Skip to content

Commit

Permalink
Committing to the open source
Browse files Browse the repository at this point in the history
  • Loading branch information
matzew committed Jul 19, 2016
1 parent f372246 commit 9072781
Show file tree
Hide file tree
Showing 63 changed files with 7,824 additions and 0 deletions.
19 changes: 19 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
lib-cov/
.project
.settings/
dist/
output/
man/
plato/
node_modules/
*~
npm*.log
fhc-debug.log
def-debug.log
.DS_Store
._.DS_Store
.idea/
dump.rdb
cov-test/
cov-unit/
coverage/
28 changes: 28 additions & 0 deletions .jshintrc
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
{
"asi" : true,
"camelcase" : false,
"bitwise" : false,
"unused" : true,
"laxbreak" : true,
"laxcomma" : true,
"curly" : false,
"eqeqeq" : true,
"evil" : true,
"forin" : false,
"immed" : true,
"latedef" : false,
"newcap" : false,
"noarg" : true,
"noempty" : true,
"nonew" : true,
"plusplus" : false,
"regexp" : true,
"undef" : false,
"strict" : false,
"sub" : true,
"trailing" : true,
"node" : true,
"maxerr" : 100,
"indent" : 2
}

48 changes: 48 additions & 0 deletions Gruntfile.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
var _ = require('underscore');

module.exports = function(grunt) {
'use strict';

function makeTestArgs(testFile) {
return ['-u exports --recursive -t 10000 ./test/setup.js', testFile].join(' ');
}

function makeUnits(testArgString) {
return [test_runner, testArgString].join(' ');
}

function makeUnitCovers(testArgString) {
return ['istanbul cover --dir cov-unit', test_runner, '--', testArgString].join(' ');
}

var tests = [ /* If updating this list of tests, also update test_win.cmd for Windows */
'./test/test_fhutils.js',
'./test/test_fhact.js',
'./test/test_fhdb.js',
'./test/test_fhfeed.js',
'./test/test_fhforms.js',
'./test/test_fhsec.js',
'./test/test_fhsession.js',
'./test/test_fhstat.js',
'./test/test_cache.js',
'./test/test_redis.js',
'./test/test_sync.js',
/*'./test/test_web.js',*/
'./test/test_fhauth.js',
'./test/test_init.js'
];
var unit_args = _.map(tests, makeTestArgs);
var test_runner = '_mocha';

// Just set shell commands for running different types of tests
grunt.initConfig({

// These are the properties that grunt-fh-build will use

unit: _.map(unit_args, makeUnits),
unit_cover: _.map(unit_args, makeUnitCovers)
});

grunt.loadNpmTasks('grunt-fh-build');
grunt.registerTask('default', ['fh:default']);
};
37 changes: 37 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
#Overview

fh-mbaas-api provides FeedHenry MBaaS APIs to Node.js cloud apps.

#Usage
fh-mbaas-api is included as standard with your cloud app code.

For custom apps, add the module via npm by running the following for the root of your app

```
npm install --save fh-mbaas-api
```

This will install the latest version of fh-mbaas-api and save the installed version in your package.json

#Documentation
Documentation for the $fh cloud API is maintained at the [FeedHenry API Docs.](http://docs.feedhenry.com/v3/api/cloud_api.html)

#Deprecated
Legacy Rhino functions have been deprecated. These are listed below - with their replacements **in bold**. All replacements listed but '$fh.web' have drop-in replacements available.

* $fh.web -> **[request](https://github.com/mikeal/request)**
* $fh.log -> **console.log**
* $fh.parse -> **JSON.parse**
* $fh.stringify **JSON.stringify**

#Tests
Before running tests do:

```
npm install
npm install -g grunt-cli
```

Then to run the tests use ```grunt fh:test```

On Windows, use ```npm run testwindows```
194 changes: 194 additions & 0 deletions lib/act.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,194 @@
var util = require('util');
var url = require('url');
var request = require('request');
var async = require('async');
var futils = require('./fhutils');
var optval = require('optval');
var fhutils;
var call;
var config;
var widget;
var cache;

module.exports = function (cfg) {
if (!cfg) {
return;
}
config = cfg;
widget = config.fhapi.widget;
fhutils = new futils(cfg);
call = require('./call')(cfg);
cache = require('./cache')(cfg);
return act();
};

/**
* description: fh.act is the server side equivilant of the client side $fh.act.
* It allows one app to call anothers endpoints which means several apps can share the same
* state information held by another app
* @param params {guid:"",endpoint:"",params:{}}
* @param cb function
*/
var act = function () {
//private set up stuff
var ERRORS = {
"InvalidCallback": "$fh.act requires the final parameter to be a function",
"InvalidArgument": "The param %s is invalid it is required to be a %s",
"MissingArgument": "The param %s is missing",
"MissingParms": "params is missing",
"InvalidGuid": "The app identifier (guid) is invalid %s . Please check it.",
"HostNotFound": "Unable to determine hostname for target app %s",
"InternalError": "Internal Error. Unable to complete lookup."
};

function validateParams(params, cb) {

if (!params || 'object' !== typeof params) {
return cb(ERRORS.MissingParms);
}
if (!params.hasOwnProperty('guid') || 'string' !== typeof params.guid) {
return cb(util.format(ERRORS.InvalidArgument, "guid", "string"));
}
if (params.guid.length !== 24) {
return cb(util.format(ERRORS.InvalidGuid, params.guid));
}
if (params.hasOwnProperty('params') && 'object' !== typeof params['params']) {
return cb(util.format(ERRORS.InvalidArgument, "params", "object"));
}
if (!(params.hasOwnProperty('path') || params.hasOwnProperty('endpoint') )) {
return cb('Either "path" or "endpoint" is required.')
}
return cb(null);
}

function doAppCall(callurl, actParams, reqParams, cb) {

callurl = url.parse(callurl);

var urlStr = url.format({
host: callurl.host,
protocol: callurl.protocol,
pathname: (actParams.path) ?
fhutils.urlPathJoin(callurl.pathname, actParams.path) :
fhutils.urlPathJoin(callurl.pathname, 'cloud', actParams.endpoint)
});

var headers = optval(actParams.headers, {});
headers["accept"] = "application/json";
headers["x-request-with"] = widget;
fhutils.addAppApiKeyHeader(headers);

var reqOpts = {
url: urlStr,
headers: headers,
method: optval(actParams.method, "POST"),
timeout: optval(actParams.timeout, 60000),
json: optval(actParams.json, true)
};

if (reqOpts.method.toLowerCase() === 'get') {
reqOpts.qs = reqParams;
} else {
reqOpts.json = reqParams;
}

request(reqOpts, function (error, response, body) {
return cb(error, body, response); // would prefer the same order as request, but maintaining backward compatibility with earlier flawed api
});

}

//return our public function
return function(params, cb) {
if ('function' !== typeof cb) throw {
name: "InvalidCallback",
message: ERRORS.InvalidCallback
};

validateParams(params, function(err) {

if (err) return cb(err);
var funcParams = params.params || {};

if (process.env.FH_SERVICE_MAP) {
// For local development, we will provide the service <==> hostname:port mapping in gruntfile.
// console.log('LOCAL service map : ', util.inspect(process.env.FH_SERVICE_MAP))
var hasUrl = false;
var serviceMap;
try {
serviceMap = JSON.parse(process.env.FH_SERVICE_MAP);
if (serviceMap[params.guid]) {
hasUrl = true;
}
else {
return cb('Unable to find mapping for guid ' + params.guid + ' in service map from FH_SERVICE_MAP environment variable');
}
} catch (e) {
return cb('Unable to parse local service map from FH_SERVICE_MAP environment variable');
}

// Do the HTTP request outside of the try catch block so that any errors are raised correctly and not swallowed by the catch block
if (hasUrl) {
doAppCall(serviceMap[params.guid], params, funcParams, cb);
}
}
else {
//For non local development we need to call core platform, sending the guid of app being called and the guid of the app doing the calling and the environment
executeAppCall(params, cb);
}
});
};


function executeAppCall(params, cb){
var hostCacheKey = params.guid + "-" + process.env.FH_ENV;
var funcParams = params.params || {};
async.waterfall([
function checkCacheForHost(callback) {
cache({"act": "load", "key": hostCacheKey}, function (err, host) {
//we ignore the error as caching miss or redis miss should not stop us making the call
callback(undefined, host);
});
},
function getHost(host, callback) {
if (host) return callback(undefined, host);
call("ide/apps/app/hosts", {
payload: {
"guid": params.guid,
"calling_guid": widget,
"env": process.env.FH_ENV
}
}, function (err, res) {
if (err) return callback(err);
//If we don't have a 200 it may mean we can't talk to core
if ('object' !== typeof res || res.status !== 200) return callback(ERRORS.InternalError);
var resData = JSON.parse(res['body']);
var callurl;

if (resData['hosts']) {
if (resData['hosts']['url']) {
// Use the generic URL if available - this caters for full lifecycle management
callurl = resData['hosts']['url'];
} else {
var live = (params.live) ? true : false;
callurl = (live) ? resData['hosts']['live-url'] : resData['hosts']['development-url'];
}
} else {
return callback(util.format(ERRORS.HostNotFound, params.guid));
}
//save the host to th cache expire after 5 mins
cache({"act":"save","value":callurl,"key": hostCacheKey,"expire": 5*60},function(){
//again we ignore the error as cache miss or redis miss should not stop us making the call
callback(undefined, callurl);
});
});
},

function callApp(host, callback) {
doAppCall(host, params, funcParams, callback);
}

], cb);
}

};
Loading

0 comments on commit 9072781

Please sign in to comment.