-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathapi.js
566 lines (519 loc) · 19.3 KB
/
api.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
/**************************************************************************
* Copyright (c) 2012 Adtec Productions, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
**************************************************************************/
var
_fs = require('fs'),
_https = require('https'),
_mime = require('mime'),
_req = require('request'),
_util = require('util'),
_URL = 'build.phonegap.com';
/******************************************************************
* Executes the GET Phonegap API call,
* and writes the output in a JSON-formatted file
******************************************************************/
var getApiData = function(token, apiCall, callback){
var
options = null;
options = {
host: _URL,
path: '/api/v1/' + apiCall + '?auth_token='+token
};
_https.get(options, function(res){
var replyData = '';
res.on('data', function(data){
replyData+= data;
});
res.on('end', function(){
if(callback instanceof Function){
callback(JSON.parse(replyData));
}
else if (callback.success instanceof Function){
callback.success(JSON.parse(replyData));
}
});
}).on('error', function(e){
console.log("AJAX Err. Message: " + e.message);
if(callback.error && (callback.error instanceof Function)){callback.error(e.message);}
});
};
/******************************************************************
* Downloads the file at the given URL and saves it in the provided
* path.
******************************************************************/
var downloadFile = function(url, outputFilepath, callback){
var successCallback = (callback instanceof Function)? callback: (callback.success instanceof Function)? callback.success: function(){};
var errCallback = (callback.error instanceof Function)? callback.error: function(){};
_req.get(url).pipe(_fs.createWriteStream(outputFilepath))
.on('error', function(e){errCallback(e.message);})
.on('close', function(){successCallback(outputFilepath);});
};
/******************************************************************
* Get a JSON-encoded representation of the authenticated user,
* as well as a listing of associated resources.
*
* This should be the starting point for applications traversing
* the PhoneGap Build API. It is aliased to
* https://build.phonegap.com/api/v1.
*
* GET https://build.phonegap.com/api/v1/me
*****************************************************************/
var _getUserData = function(token, callback){
getApiData(token, 'me', callback);
};
/******************************************************************
* Get a JSON-encoded representation of the authenticated user's
* apps.
*
* API clients can follow the link attribute for each app to get
* further details, including the associated signing keys and
* collaborators.
*
* GET https://build.phonegap.com/api/v1/apps
*****************************************************************/
var _getAppsData = function(token, callback){
getApiData(token, 'apps', callback);
};
/******************************************************************
* Get a JSON-encoded representation of a particular app, if the
* authenticated user has permission to access it.
*
* In addition to the fields provided in the list of all apps,
* this detail view includes:
*
* keys: all of the keys that the app is currently being built with.
* This will include the owner's default key for a platform,
* if selected
*
* collaborators: each person who has access to this app, along
* with their role, if the authenticated user is the owner of
* the app. Collaborators who are registered with PhoneGap Build
* are listed under active; collaborators you have invited who
* have not yet created an account are listed as pending.
*
* GET https://build.phonegap.com/api/v1/apps/:id
*****************************************************************/
var _getAppDataById = function(token, appId, callback){
getApiData(token, 'apps/' + appId, callback);
};
/******************************************************************
* Get a JSON-encoded list of all the signing keys associated with
* your account.
*
* This returns a short listing of all the associated keys--it's
* very similar to the list you'll see when requesting /api/v1/me
*
* GET https://build.phonegap.com/api/v1/keys
*****************************************************************/
var _getKeysData = function(token, callback){
getApiData(token, 'keys', callback);
};
/******************************************************************
* Get a JSON-encoded list of all the signing keys associated with
* your account, for a specific platform. That platform can be one
* of ios, android, or blackberry.
*
* GET https://build.phonegap.com/api/v1/keys/:platform
*****************************************************************/
var _getPlatformKeys = function(platform){
getApiData('keys/' + platform, METADATA_DIR + '/keys_' + platform + '.json');
};
/******************************************************************
* Get a JSON-encoded representation of a single signing key.
*
* GET https://build.phonegap.com/api/v1/keys/:platform/:appId
*****************************************************************/
var _getPlatformKeyById = function(platform, appId){
getApiData('keys/' + platform + '/' + appId, METADATA_DIR + '/keys_' + platform + '_' + appId + '.json');
};
/******************************************************************
* Download the app package for the given platform; available
* platforms are android, blackberry, ios, symbian, webos and
* winphone.
*
* In the successful case, this API method will return a 302
* redirect to the application binary - the actual body of the
* response will point to the resource's correct location:
*
* If using the optional argument for the download location,
* please ensure that you are using the right extension for
* the platform you are downloading.
*
* apk for Android
* ipa for iOS
* ipk for webOS
* jad for unsigned BlackBerry builds; zip if you've uploaded your BlackBerry signing keys
* wgz for Symbian
* xap for Windows Phone
*
* GET https://build.phonegap.com/api/v1/apps/:id/:platform
*****************************************************************/
var _downloadApp = function(token, appId, platform, outputFilepath, callback){
var url= 'https://' + _URL + '/api/v1/apps/' + appId + '/' + platform + '?auth_token='+token;
console.info("\n\nStarting Download...");
downloadFile(url, outputFilepath, callback);
};
/*****************************************************************
* Get the main icon associated with an app - this is either the
* biggest icon specified in your config.xml file, or an icon you
* have uploaded through the API or the PhoneGap Build web interface.
*
* GET https://build.phonegap.com/api/v1/apps/:id/:icon
*****************************************************************/
var _downloadIcon = function(token, appId, outputFilepath, callback){
var url= 'https://' + _URL + '/api/v1/apps/' + appId + '/icon';
downloadFile(url, outputFilepath, callback);
};
/************************************
* Init Write API
*/
var encodeFieldHeader = function(boundary, name, value){
var result = [
"--" + boundary + "\r\n",
"Content-Disposition: form-data; name=\"",
"" + name + "\"\r\n\r\n" + value + "\r\n"
].join("");
return result;
};
var encodeFileHeader = function(boundary, type, name, filename){
var result = [
"--" + boundary + "\r\n",
"Content-Disposition: form-data; name=\"" + name + "\"; filename=\"" + filename + "\"\r\n",
"Content-Type: " + type + "\r\n\r\n"
].join("");
return result;
};
var postMultipart = function(token, postData, boundary, apiCall, callback){
var
len = 0, i=0,
options = null,
request = null;
for( i=0; i<postData.length; i++){
len += postData[i].length;
}
options = {
host: _URL,
path: '/api/v1/' + apiCall + '?auth_token='+token,
method: 'POST',
headers:{
'Content-Type': 'multipart/form-data; boundary=' + boundary,
'Content-Length': len
}
};
request = _https.request(options, function(response){
var replyData = '';
response.on('data', function(chunk){
replyData+= chunk;
});
response.on('end', function(){
if(callback instanceof Function){
callback(replyData);
}
else if(callback.success instanceof Function){
callback.success(replyData);
}
return;
});
response.on('error', function(e){
if(callback.error instanceof Function){
callback.error(e.message, response.statusCode);
}
return;
});
});
for( i=0; i<postData.length; i++){
request.write(postData[i]);
}
request.end();
};
var initMultipartUpload = function(token, inputFile, dataObj, apiCall, fieldName, callback){
var
boundary = 'bound' + Math.random(),
postData = [],
fileReader = null,
fileContents = '';
if(dataObj){postData.push(new Buffer(encodeFieldHeader(boundary, "data", JSON.stringify(dataObj)), 'ascii'));}
postData.push(new Buffer(encodeFileHeader(boundary, _mime.lookup(inputFile), fieldName, inputFile), 'ascii'));
fileReader = _fs.createReadStream(inputFile, {encoding: 'binary'});
fileReader.on('data', function(fileData){ fileContents+= fileData;});
fileReader.on('end', function(){
postData.push(new Buffer(fileContents, 'binary'));
postData.push(new Buffer("\r\n--" + boundary + "--", 'ascii'));
postMultipart(token, postData, boundary, apiCall, callback);
});
};
var initFileUpload = function(token, inputFile, apiCall, callback){
var apiPath = '/api/v1/' + apiCall + '?auth_token=' + token;
try {
_fs.createReadStream(inputFile).pipe(_req({
method: 'PUT',
url: 'https://build.phonegap.com' + apiPath
}, function(err, res, body){
if(err) {
_error(err);
} else if (res && (res.statusCode < 200 || res.statusCode >= 300)) {
_error(new Error(res.statusCode.toString() || body));
}
}))
.on('error', _error);
} catch(e) {
_error(e);
}
function _error(e) {
if(callback.error instanceof Function) {
if (e instanceof Error) {
callback.error(e);
} else if (e.message) {
callback.error(new Error(e.message));
} else {
callback.error(new Error(e.toString()));
}
}
}
};
var initWebFormUpload = function(token, dataObj, apiCall, callback) {
var apiPath = '/api/v1/' + apiCall + '?auth_token=' + token;
console.log("apiPath: " + apiPath);
try {
_req({
method: 'PUT',
url: 'https://build.phonegap.com' + apiPath,
form: {data: JSON.stringify(dataObj)}
}, function(err, res, body){
console.log("err: " + err + ", body:" + _util.inspect(body));
if(err) {
_error(err);
} else if (res && (res.statusCode < 200 || res.statusCode >= 300)) {
_error(new Error(res.statusCode.toString() || body));
} else {
if(callback instanceof Function){
callback(body);
} else if(callback.success instanceof Function){
callback.success(body);
}
}
});
} catch(e) {
_error(e);
}
function _error(e) {
if(callback.error instanceof Function) {
if (e instanceof Error) {
callback.error(e);
} else if (e.message) {
callback.error(new Error(e.message));
} else {
callback.error(new Error(e.toString()));
}
}
}
};
/******************************************************************
*
* Create a new File-based App.
*
* Required properties of dataObj:
*
* title: You must specify a title for your app - if a title is also
* specified in a config.xml in your package, the one in the
* config.xml file will take preference.
* create_method: use "file" for this method
*
* Optional properties of dataObj:
*
* package: Sets the package identifier for your app. This can also be
* done after creation, or in your config.xml file.
* Defaults to com.phonegap.www
* version: Sets the version of your app. This can also be done after
* creation, or in your config.xml file. Defaults to 0.0.1
* description: Sets the description for your app. This can also be
* done after creation, or in your config.xml file.
* Defaults to empty.
* debug: Builds your app in debug mode. Defaults to false.
* keys: Set the signing keys to use for each platform you wish to sign.
* private: Whether your app can be publicly downloaded. Defaults to
* true during beta period; will default to false once the
* beta period is complete
* phonegap_version: Which version of PhoneGap your app uses. See
* config.xml for details on which are supported,
* and which one is currently the default
*
* File-backed applications
*
* To create a file-backed application, set the create_method parameter
* to file, and include a zip file, a tar.gz file, or an index.html
* file in the multipart body of your post, with the parameter name file.
*
* POST https://build.phonegap.com/api/v1/apps
*****************************************************************/
var _createFileBasedApp = function(token, inputFile, dataObj, callback){
initMultipartUpload(token, inputFile, dataObj, 'apps', "file", callback);
};
/******************************************************************
* Updating a file-based application
*
* If the application has been created from a file upload, you
* can include a new index.html, zip file, or tar.gz file as the
* file parameter in your request to update the contents.
*
* Optional properties of dataObj:
*
* - keys: Set the signing keys to use for each platform you wish to sign.
*
* PUT https://build.phonegap.com/api/v1/apps/:id
*****************************************************************/
var _updateFileBasedApp = function(token, inputFile, appId, dataObj, callback){
var doUpload = initFileUpload.bind(null, token, inputFile, 'apps/' + appId, callback);
if (dataObj.keys) {
_uploadAppKeys(token, appId, dataObj, {
success: doUpload,
error: callback.error
});
} else {
doUpload();
}
};
/**
* PUT https://build.phonegap.com/api/v1/apps/:id
*
* Set the signing keys (and unlock them if the credentials are part
* of the request) for the given application.
*
* Mandatory parameters of dataObj:
*
* - keys
*/
var _uploadAppKeys = function(token, appId, dataObj, callback) {
initWebFormUpload(token, {keys: dataObj.keys}, 'apps/' + appId, callback);
};
/******************************************************************
* POST https://build.phonegap.com/api/v1/apps/:id/:icon
*
* Sets an icon file for a given app. Send a png file as the icon
* parameter in your post.
* If you want to have multiple icons for different resolutions, you
* should not use this API method. Instead, include the different
* icon files in your application package and specify their use
* in your config.xml file.
*
* The response will have a 201 created status, and the
* application will be queued for building.
******************************************************************/
var _uploadAppIcon = function(token, appId, inputFile, callback){
initMultipartUpload(token, inputFile, null, 'apps/' + appId + "/icon", "icon", callback);
};
/******************************************************************
*
* DELETE https://build.phonegap.com/api/v1/apps/:id
*
* Delete your application from PhoneGap Build - will return either
* a 202 (accepted) status, or 404 (if the app cannot be found).
*****************************************************************/
var _deleteFileBasedApp = function(token, appId, callback){
var options = {
url : 'https://'+_URL+"/api/v1/apps/" + appId + '?auth_token=' + token
};
_req.del(options, function (error, response, body) {
if(error){
if(callback instanceof Function){
callback("Error deleting app " + appId);
}
else if(callback.error instanceof Function){
callback.error("Error deleting app " + appId);
}
}
else{
if(callback instanceof Function){
callback(body);
}
else if(callback.success instanceof Function){
callback.success(body);
}
}
});
};
/******************************************************************
* POST https://build.phonegap.com/api/v1/apps/:id/build
*
* Queue new builds for a specified app. The older builds will be
* discarded, while new ones are queued.
* The builds will use the most current app contents, as well as
* the selected signing keys. The response will have a 202 (accepted)
* status.
*
* To choose which platforms to build, include those as a JSON encoded
* parameter in your post
*
* Once the builds are queued, you will want to watch the results of
* GET /api/v1/apps/:id to see when each platform's status changes
* from pending (to complete or error).
*
******************************************************************/
//rebuildApp = function(token, appId, dataObj, callback){
//
//},
var _createAuthToken = function(rawCredentials, callback){
var
auth = "Basic " + new Buffer(rawCredentials).toString("base64"),
options = {
url : 'https://'+_URL+"/token",
headers : { "Authorization" : auth }
};
_req.post(options, function (error, response, body) {
response = response || {};
if((error!==null) || (response.statusCode!=200))
{
var errStr = "AJAX error. Your request could not be completed. Please verify your login credentials and network access. statusCode: " + response.statusCode;
if(body && JSON.parse(body) && JSON.parse(body).error){errStr +="\n" + JSON.parse(body).error;}
if(callback.error instanceof Function){callback.error(errStr, response.statusCode);}
return;
}
if(callback instanceof Function){
callback(JSON.parse(body).token);
}
else if(callback.success instanceof Function){
callback.success(JSON.parse(body).token);
}
});
};
/******************************************************************
* Module Public Members
*****************************************************************/
module.exports = {
//Read API
getUserData: _getUserData,
getAppsData: _getAppsData,
getAppDataById: _getAppDataById,
getKeysData: _getKeysData,
getPlatformKeys:_getPlatformKeys,
getPlatformKeyById: _getPlatformKeyById,
downloadApp: _downloadApp,
downloadIcon:_downloadIcon,
//Write API
createFileBasedApp:_createFileBasedApp,
updateFileBasedApp:_updateFileBasedApp,
uploadAppKeys: _uploadAppKeys,
uploadAppIcon: _uploadAppIcon,
deleteFileBasedApp: _deleteFileBasedApp,
createAuthToken: _createAuthToken
};
/*
* Local Variables:
* mode: js2
* js2-basic-offset: 3
* indent-tabs-mode: nil
* End:
*/