mirror of
https://github.com/apache/cordova-android.git
synced 2025-02-27 04:42:51 +08:00
Merge remote-tracking branch 'upstream/master' into volumebtns
This commit is contained in:
commit
cb61e90148
@ -23,7 +23,7 @@
|
|||||||
<meta http-equiv="Content-type" content="text/html; charset=utf-8">
|
<meta http-equiv="Content-type" content="text/html; charset=utf-8">
|
||||||
<title>PhoneGap</title>
|
<title>PhoneGap</title>
|
||||||
<link rel="stylesheet" href="master.css" type="text/css" media="screen" title="no title">
|
<link rel="stylesheet" href="master.css" type="text/css" media="screen" title="no title">
|
||||||
<script type="text/javascript" charset="utf-8" src="cordova-1.7.0.js"></script>
|
<script type="text/javascript" charset="utf-8" src="cordova-1.8.0rc1.js"></script>
|
||||||
<script type="text/javascript" charset="utf-8" src="main.js"></script>
|
<script type="text/javascript" charset="utf-8" src="main.js"></script>
|
||||||
|
|
||||||
</head>
|
</head>
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// commit 4a4ba9985c920850fe3f229abc60de984e196ab5
|
// commit 95f199e1c207dc89b84e79a9a7b27d6a3cc8fe14
|
||||||
|
|
||||||
// File generated at :: Fri May 18 2012 13:43:11 GMT-0700 (PDT)
|
// File generated at :: Thu May 24 2012 21:36:17 GMT-0400 (EDT)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Licensed to the Apache Software Foundation (ASF) under one
|
Licensed to the Apache Software Foundation (ASF) under one
|
||||||
@ -714,6 +714,9 @@ module.exports = {
|
|||||||
children: {
|
children: {
|
||||||
exec: {
|
exec: {
|
||||||
path: 'cordova/exec'
|
path: 'cordova/exec'
|
||||||
|
},
|
||||||
|
logger: {
|
||||||
|
path: 'cordova/plugin/logger'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -1195,7 +1198,7 @@ for (var key in Camera) {
|
|||||||
* Gets a picture from source defined by "options.sourceType", and returns the
|
* Gets a picture from source defined by "options.sourceType", and returns the
|
||||||
* image as defined by the "options.destinationType" option.
|
* image as defined by the "options.destinationType" option.
|
||||||
|
|
||||||
* The defaults are sourceType=CAMERA and destinationType=FILE_URL.
|
* The defaults are sourceType=CAMERA and destinationType=FILE_URI.
|
||||||
*
|
*
|
||||||
* @param {Function} successCallback
|
* @param {Function} successCallback
|
||||||
* @param {Function} errorCallback
|
* @param {Function} errorCallback
|
||||||
@ -2141,7 +2144,7 @@ Entry.prototype.toURL = function() {
|
|||||||
Entry.prototype.toURI = function(mimeType) {
|
Entry.prototype.toURI = function(mimeType) {
|
||||||
console.log("DEPRECATED: Update your code to use 'toURL'");
|
console.log("DEPRECATED: Update your code to use 'toURL'");
|
||||||
// fullPath attribute contains the full URI
|
// fullPath attribute contains the full URI
|
||||||
return this.fullPath;
|
return this.toURL();
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -4569,6 +4572,177 @@ var exec = require('cordova/exec'),
|
|||||||
module.exports = compass;
|
module.exports = compass;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// file: lib/common/plugin/console-via-logger.js
|
||||||
|
define("cordova/plugin/console-via-logger", function(require, exports, module) {
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
var logger = require("cordova/plugin/logger");
|
||||||
|
var utils = require("cordova/utils");
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// object that we're exporting
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
var console = module.exports;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// copy of the original console object
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
var WinConsole = window.console;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// whether to use the logger
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
var UseLogger = false;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// Timers
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
var Timers = {};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// used for unimplemented methods
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
function noop() {}
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// used for unimplemented methods
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.useLogger = function (value) {
|
||||||
|
if (arguments.length) UseLogger = !!value;
|
||||||
|
|
||||||
|
if (UseLogger) {
|
||||||
|
if (logger.useConsole()) {
|
||||||
|
throw new Error("console and logger are too intertwingly");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return UseLogger;
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.log = function() {
|
||||||
|
if (logger.useConsole()) return;
|
||||||
|
logger.log.apply(logger, [].slice.call(arguments));
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.error = function() {
|
||||||
|
if (logger.useConsole()) return;
|
||||||
|
logger.error.apply(logger, [].slice.call(arguments));
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.warn = function() {
|
||||||
|
if (logger.useConsole()) return;
|
||||||
|
logger.warn.apply(logger, [].slice.call(arguments));
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.info = function() {
|
||||||
|
if (logger.useConsole()) return;
|
||||||
|
logger.info.apply(logger, [].slice.call(arguments));
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.debug = function() {
|
||||||
|
if (logger.useConsole()) return;
|
||||||
|
logger.debug.apply(logger, [].slice.call(arguments));
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.assert = function(expression) {
|
||||||
|
if (expression) return;
|
||||||
|
|
||||||
|
var message = utils.vformat(arguments[1], [].slice.call(arguments, 2));
|
||||||
|
console.log("ASSERT: " + message);
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.clear = function() {};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.dir = function(object) {
|
||||||
|
console.log("%o", object);
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.dirxml = function(node) {
|
||||||
|
console.log(node.innerHTML);
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.trace = noop;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.group = console.log;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.groupCollapsed = console.log;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.groupEnd = noop;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.time = function(name) {
|
||||||
|
Timers[name] = new Date().valueOf();
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.timeEnd = function(name) {
|
||||||
|
var timeStart = Timers[name];
|
||||||
|
if (!timeStart) {
|
||||||
|
console.warn("unknown timer: " + name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var timeElapsed = new Date().valueOf() - timeStart;
|
||||||
|
console.log(name + ": " + timeElapsed + "ms");
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.timeStamp = noop;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.profile = noop;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.profileEnd = noop;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.count = noop;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.exception = console.log;
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
console.table = function(data, columns) {
|
||||||
|
console.log("%o", data);
|
||||||
|
};
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// return a new function that calls both functions passed as args
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
function wrapperedOrigCall(orgFunc, newFunc) {
|
||||||
|
return function() {
|
||||||
|
var args = [].slice.call(arguments);
|
||||||
|
try { orgFunc.apply(WinConsole, args); } catch (e) {}
|
||||||
|
try { newFunc.apply(console, args); } catch (e) {}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// For every function that exists in the original console object, that
|
||||||
|
// also exists in the new console object, wrap the new console method
|
||||||
|
// with one that calls both
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
for (var key in console) {
|
||||||
|
if (typeof WinConsole[key] == "function") {
|
||||||
|
console[key] = wrapperedOrigCall(WinConsole[key], console[key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
// file: lib/common/plugin/contacts.js
|
// file: lib/common/plugin/contacts.js
|
||||||
define("cordova/plugin/contacts", function(require, exports, module) {
|
define("cordova/plugin/contacts", function(require, exports, module) {
|
||||||
var exec = require('cordova/exec'),
|
var exec = require('cordova/exec'),
|
||||||
@ -4830,6 +5004,233 @@ module.exports = geolocation;
|
|||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// file: lib/common/plugin/logger.js
|
||||||
|
define("cordova/plugin/logger", function(require, exports, module) {
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// The logger module exports the following properties/functions:
|
||||||
|
//
|
||||||
|
// LOG - constant for the level LOG
|
||||||
|
// ERROR - constant for the level ERROR
|
||||||
|
// WARN - constant for the level WARN
|
||||||
|
// INFO - constant for the level INFO
|
||||||
|
// DEBUG - constant for the level DEBUG
|
||||||
|
// logLevel() - returns current log level
|
||||||
|
// logLevel(value) - sets and returns a new log level
|
||||||
|
// useConsole() - returns whether logger is using console
|
||||||
|
// useConsole(value) - sets and returns whether logger is using console
|
||||||
|
// log(message,...) - logs a message at level LOG
|
||||||
|
// error(message,...) - logs a message at level ERROR
|
||||||
|
// warn(message,...) - logs a message at level WARN
|
||||||
|
// info(message,...) - logs a message at level INFO
|
||||||
|
// debug(message,...) - logs a message at level DEBUG
|
||||||
|
// logLevel(level,message,...) - logs a message specified level
|
||||||
|
//
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
var logger = exports;
|
||||||
|
|
||||||
|
var exec = require('cordova/exec');
|
||||||
|
var utils = require('cordova/utils');
|
||||||
|
|
||||||
|
var UseConsole = true;
|
||||||
|
var Queued = [];
|
||||||
|
var DeviceReady = false;
|
||||||
|
var CurrentLevel;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logging levels
|
||||||
|
*/
|
||||||
|
|
||||||
|
var Levels = [
|
||||||
|
"LOG",
|
||||||
|
"ERROR",
|
||||||
|
"WARN",
|
||||||
|
"INFO",
|
||||||
|
"DEBUG"
|
||||||
|
];
|
||||||
|
|
||||||
|
/*
|
||||||
|
* add the logging levels to the logger object and
|
||||||
|
* to a separate levelsMap object for testing
|
||||||
|
*/
|
||||||
|
|
||||||
|
var LevelsMap = {};
|
||||||
|
for (var i=0; i<Levels.length; i++) {
|
||||||
|
var level = Levels[i];
|
||||||
|
LevelsMap[level] = i;
|
||||||
|
logger[level] = level;
|
||||||
|
}
|
||||||
|
|
||||||
|
CurrentLevel = LevelsMap.WARN;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Getter/Setter for the logging level
|
||||||
|
*
|
||||||
|
* Returns the current logging level.
|
||||||
|
*
|
||||||
|
* When a value is passed, sets the logging level to that value.
|
||||||
|
* The values should be one of the following constants:
|
||||||
|
* logger.LOG
|
||||||
|
* logger.ERROR
|
||||||
|
* logger.WARN
|
||||||
|
* logger.INFO
|
||||||
|
* logger.DEBUG
|
||||||
|
*
|
||||||
|
* The value used determines which messages get printed. The logging
|
||||||
|
* values above are in order, and only messages logged at the logging
|
||||||
|
* level or above will actually be displayed to the user. Eg, the
|
||||||
|
* default level is WARN, so only messages logged with LOG, ERROR, or
|
||||||
|
* WARN will be displayed; INFO and DEBUG messages will be ignored.
|
||||||
|
*/
|
||||||
|
logger.level = function (value) {
|
||||||
|
if (arguments.length) {
|
||||||
|
if (LevelsMap[value] === null) {
|
||||||
|
throw new Error("invalid logging level: " + value);
|
||||||
|
}
|
||||||
|
CurrentLevel = LevelsMap[value];
|
||||||
|
}
|
||||||
|
|
||||||
|
return Levels[CurrentLevel];
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Getter/Setter for the useConsole functionality
|
||||||
|
*
|
||||||
|
* When useConsole is true, the logger will log via the
|
||||||
|
* browser 'console' object. Otherwise, it will use the
|
||||||
|
* native Logger plugin.
|
||||||
|
*/
|
||||||
|
logger.useConsole = function (value) {
|
||||||
|
if (arguments.length) UseConsole = !!value;
|
||||||
|
|
||||||
|
if (UseConsole) {
|
||||||
|
if (typeof console == "undefined") {
|
||||||
|
throw new Error("global console object is not defined");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof console.log != "function") {
|
||||||
|
throw new Error("global console object does not have a log function");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof console.useLogger == "function") {
|
||||||
|
if (console.useLogger()) {
|
||||||
|
throw new Error("console and logger are too intertwingly");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return UseConsole;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logs a message at the LOG level.
|
||||||
|
*
|
||||||
|
* Parameters passed after message are used applied to
|
||||||
|
* the message with utils.format()
|
||||||
|
*/
|
||||||
|
logger.log = function(message) { logWithArgs("LOG", arguments); };
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logs a message at the ERROR level.
|
||||||
|
*
|
||||||
|
* Parameters passed after message are used applied to
|
||||||
|
* the message with utils.format()
|
||||||
|
*/
|
||||||
|
logger.error = function(message) { logWithArgs("ERROR", arguments); };
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logs a message at the WARN level.
|
||||||
|
*
|
||||||
|
* Parameters passed after message are used applied to
|
||||||
|
* the message with utils.format()
|
||||||
|
*/
|
||||||
|
logger.warn = function(message) { logWithArgs("WARN", arguments); };
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logs a message at the INFO level.
|
||||||
|
*
|
||||||
|
* Parameters passed after message are used applied to
|
||||||
|
* the message with utils.format()
|
||||||
|
*/
|
||||||
|
logger.info = function(message) { logWithArgs("INFO", arguments); };
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logs a message at the DEBUG level.
|
||||||
|
*
|
||||||
|
* Parameters passed after message are used applied to
|
||||||
|
* the message with utils.format()
|
||||||
|
*/
|
||||||
|
logger.debug = function(message) { logWithArgs("DEBUG", arguments); };
|
||||||
|
|
||||||
|
// log at the specified level with args
|
||||||
|
function logWithArgs(level, args) {
|
||||||
|
args = [level].concat([].slice.call(args));
|
||||||
|
logger.logLevel.apply(logger, args);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logs a message at the specified level.
|
||||||
|
*
|
||||||
|
* Parameters passed after message are used applied to
|
||||||
|
* the message with utils.format()
|
||||||
|
*/
|
||||||
|
logger.logLevel = function(level, message /* , ... */) {
|
||||||
|
// format the message with the parameters
|
||||||
|
var formatArgs = [].slice.call(arguments, 2);
|
||||||
|
message = utils.vformat(message, formatArgs);
|
||||||
|
|
||||||
|
if (LevelsMap[level] === null) {
|
||||||
|
throw new Error("invalid logging level: " + level);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (LevelsMap[level] > CurrentLevel) return;
|
||||||
|
|
||||||
|
// queue the message if not yet at deviceready
|
||||||
|
if (!DeviceReady && !UseConsole) {
|
||||||
|
Queued.push([level, message]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// if not using the console, use the native logger
|
||||||
|
if (!UseConsole) {
|
||||||
|
exec(null, null, "Logger", "logLevel", [level, message]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// make sure console is not using logger
|
||||||
|
if (console.__usingCordovaLogger) {
|
||||||
|
throw new Error("console and logger are too intertwingly");
|
||||||
|
}
|
||||||
|
|
||||||
|
// log to the console
|
||||||
|
switch (level) {
|
||||||
|
case logger.LOG: console.log(message); break;
|
||||||
|
case logger.ERROR: console.log("ERROR: " + message); break;
|
||||||
|
case logger.WARN: console.log("WARN: " + message); break;
|
||||||
|
case logger.INFO: console.log("INFO: " + message); break;
|
||||||
|
case logger.DEBUG: console.log("DEBUG: " + message); break;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// when deviceready fires, log queued messages
|
||||||
|
logger.__onDeviceReady = function() {
|
||||||
|
if (DeviceReady) return;
|
||||||
|
|
||||||
|
DeviceReady = true;
|
||||||
|
|
||||||
|
for (var i=0; i<Queued.length; i++) {
|
||||||
|
var messageArgs = Queued[i];
|
||||||
|
logger.logLevel(messageArgs[0], messageArgs[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
Queued = null;
|
||||||
|
};
|
||||||
|
|
||||||
|
// add a deviceready event to log queued messages
|
||||||
|
document.addEventListener("deviceready", logger.__onDeviceReady, false);
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
// file: lib/common/plugin/network.js
|
// file: lib/common/plugin/network.js
|
||||||
define("cordova/plugin/network", function(require, exports, module) {
|
define("cordova/plugin/network", function(require, exports, module) {
|
||||||
var exec = require('cordova/exec'),
|
var exec = require('cordova/exec'),
|
||||||
@ -5018,10 +5419,16 @@ module.exports = function(uri, successCallback, errorCallback) {
|
|||||||
errorCallback(new FileError(error));
|
errorCallback(new FileError(error));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
// sanity check for 'not:valid:filename'
|
||||||
|
if(!uri || uri.split(":").length > 2) {
|
||||||
|
setTimeout( function() {
|
||||||
|
fail(FileError.ENCODING_ERR);
|
||||||
|
},0);
|
||||||
|
return;
|
||||||
|
}
|
||||||
// if successful, return either a file or directory entry
|
// if successful, return either a file or directory entry
|
||||||
var success = function(entry) {
|
var success = function(entry) {
|
||||||
var result;
|
var result;
|
||||||
|
|
||||||
if (entry) {
|
if (entry) {
|
||||||
if (typeof successCallback === 'function') {
|
if (typeof successCallback === 'function') {
|
||||||
// create appropriate Entry object
|
// create appropriate Entry object
|
||||||
@ -5058,6 +5465,68 @@ var splashscreen = {
|
|||||||
module.exports = splashscreen;
|
module.exports = splashscreen;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// file: lib/common/plugin/widget.js
|
||||||
|
define("cordova/plugin/widget", function(require, exports, module) {
|
||||||
|
var exec = require('cordova/exec'),
|
||||||
|
cordova = require('cordova'),
|
||||||
|
channel = require('cordova/channel');
|
||||||
|
|
||||||
|
var Widget = function () {
|
||||||
|
this.author = null;
|
||||||
|
this.description = null;
|
||||||
|
this.name = null;
|
||||||
|
this.shortName = null;
|
||||||
|
this.version = null;
|
||||||
|
this.id = null;
|
||||||
|
this.authorEmail = null;
|
||||||
|
this.authorHref = null;
|
||||||
|
this._firstRun = true;
|
||||||
|
|
||||||
|
var me = this;
|
||||||
|
|
||||||
|
channel.onCordovaReady.subscribeOnce(function() {
|
||||||
|
me.getInfo(function (info) {
|
||||||
|
me.author = info.author;
|
||||||
|
me.description = info.description;
|
||||||
|
me.name = info.name;
|
||||||
|
me.shortName = info.shortName;
|
||||||
|
me.version = info.version;
|
||||||
|
me.id = info.id;
|
||||||
|
me.authorEmail = info.authorEmail;
|
||||||
|
me.authorHref = info.authorHref;
|
||||||
|
|
||||||
|
// should only fire this once
|
||||||
|
if (me._firstRun) {
|
||||||
|
me._firstRun = false;
|
||||||
|
channel.onCordovaAppInfoReady.fire();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
function (e) {
|
||||||
|
// If we can't get the network info we should still tell Cordova
|
||||||
|
// to fire the deviceready event.
|
||||||
|
if (me._firstRun) {
|
||||||
|
me._firstRun = false;
|
||||||
|
channel.onCordovaAppInfoReady.fire();
|
||||||
|
}
|
||||||
|
console.log("Error initializing Widget: " + e);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get connection info
|
||||||
|
*
|
||||||
|
* @param {Function} successCallback The function to call when the Connection data is available
|
||||||
|
* @param {Function} errorCallback The function to call when there is an error getting the Connection data. (OPTIONAL)
|
||||||
|
*/
|
||||||
|
Widget.prototype.getInfo = function (successCallback, errorCallback) {
|
||||||
|
// Get info
|
||||||
|
exec(successCallback, errorCallback, "Widget", "getApplicationInfo", []);
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = new Widget();
|
||||||
|
});
|
||||||
|
|
||||||
// file: lib/common/utils.js
|
// file: lib/common/utils.js
|
||||||
define("cordova/utils", function(require, exports, module) {
|
define("cordova/utils", function(require, exports, module) {
|
||||||
var utils = exports;
|
var utils = exports;
|
||||||
@ -5159,6 +5628,16 @@ utils.alert = function(msg) {
|
|||||||
/**
|
/**
|
||||||
* Formats a string and arguments following it ala sprintf()
|
* Formats a string and arguments following it ala sprintf()
|
||||||
*
|
*
|
||||||
|
* see utils.vformat() for more information
|
||||||
|
*/
|
||||||
|
utils.format = function(formatString /* ,... */) {
|
||||||
|
var args = [].slice.call(arguments, 1);
|
||||||
|
return utils.vformat(formatString, args);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Formats a string and arguments following it ala vsprintf()
|
||||||
|
*
|
||||||
* format chars:
|
* format chars:
|
||||||
* %j - format arg as JSON
|
* %j - format arg as JSON
|
||||||
* %o - format arg as JSON
|
* %o - format arg as JSON
|
||||||
@ -5170,14 +5649,13 @@ utils.alert = function(msg) {
|
|||||||
* for rationale, see FireBug's Console API:
|
* for rationale, see FireBug's Console API:
|
||||||
* http://getfirebug.com/wiki/index.php/Console_API
|
* http://getfirebug.com/wiki/index.php/Console_API
|
||||||
*/
|
*/
|
||||||
utils.format = function(formatString /* ,... */) {
|
utils.vformat = function(formatString, args) {
|
||||||
if (formatString === null || formatString === undefined) return "";
|
if (formatString === null || formatString === undefined) return "";
|
||||||
if (arguments.length == 1) return formatString.toString();
|
if (arguments.length == 1) return formatString.toString();
|
||||||
|
|
||||||
var pattern = /(.*?)%(.)(.*)/;
|
var pattern = /(.*?)%(.)(.*)/;
|
||||||
var rest = formatString.toString();
|
var rest = formatString.toString();
|
||||||
var result = [];
|
var result = [];
|
||||||
var args = [].slice.call(arguments,1);
|
|
||||||
|
|
||||||
while (args.length) {
|
while (args.length) {
|
||||||
var arg = args.shift();
|
var arg = args.shift();
|
||||||
|
@ -19,7 +19,7 @@
|
|||||||
<html>
|
<html>
|
||||||
<head>
|
<head>
|
||||||
<title></title>
|
<title></title>
|
||||||
<script src="cordova-1.7.0.js"></script>
|
<script src="cordova-1.8.0rc1.js"></script>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
|
|
||||||
|
@ -256,7 +256,9 @@ public class CameraLauncher extends Plugin {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return Bitmap.createScaledBitmap(bitmap, newWidth, newHeight, true);
|
Bitmap retval = Bitmap.createScaledBitmap(bitmap, newWidth, newHeight, true);
|
||||||
|
bitmap.recycle();
|
||||||
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -38,7 +38,7 @@ import android.telephony.TelephonyManager;
|
|||||||
public class Device extends Plugin {
|
public class Device extends Plugin {
|
||||||
public static final String TAG = "Device";
|
public static final String TAG = "Device";
|
||||||
|
|
||||||
public static String cordovaVersion = "1.7.0"; // Cordova version
|
public static String cordovaVersion = "1.8.0rc1"; // Cordova version
|
||||||
public static String platform = "Android"; // Device OS
|
public static String platform = "Android"; // Device OS
|
||||||
public static String uuid; // Device UUID
|
public static String uuid; // Device UUID
|
||||||
|
|
||||||
|
@ -27,6 +27,7 @@ import java.io.FileOutputStream;
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.net.HttpURLConnection;
|
import java.net.HttpURLConnection;
|
||||||
|
import java.net.MalformedURLException;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.security.cert.CertificateException;
|
import java.security.cert.CertificateException;
|
||||||
import java.security.cert.X509Certificate;
|
import java.security.cert.X509Certificate;
|
||||||
@ -35,7 +36,6 @@ import java.util.Iterator;
|
|||||||
import javax.net.ssl.HostnameVerifier;
|
import javax.net.ssl.HostnameVerifier;
|
||||||
import javax.net.ssl.HttpsURLConnection;
|
import javax.net.ssl.HttpsURLConnection;
|
||||||
import javax.net.ssl.SSLContext;
|
import javax.net.ssl.SSLContext;
|
||||||
import javax.net.ssl.SSLException;
|
|
||||||
import javax.net.ssl.SSLSession;
|
import javax.net.ssl.SSLSession;
|
||||||
import javax.net.ssl.SSLSocketFactory;
|
import javax.net.ssl.SSLSocketFactory;
|
||||||
import javax.net.ssl.TrustManager;
|
import javax.net.ssl.TrustManager;
|
||||||
@ -57,7 +57,7 @@ public class FileTransfer extends Plugin {
|
|||||||
private static final String LOG_TAG = "FileTransfer";
|
private static final String LOG_TAG = "FileTransfer";
|
||||||
private static final String LINE_START = "--";
|
private static final String LINE_START = "--";
|
||||||
private static final String LINE_END = "\r\n";
|
private static final String LINE_END = "\r\n";
|
||||||
private static final String BOUNDRY = "*****";
|
private static final String BOUNDARY = "*****";
|
||||||
|
|
||||||
public static int FILE_NOT_FOUND_ERR = 1;
|
public static int FILE_NOT_FOUND_ERR = 1;
|
||||||
public static int INVALID_URL_ERR = 2;
|
public static int INVALID_URL_ERR = 2;
|
||||||
@ -82,152 +82,54 @@ public class FileTransfer extends Plugin {
|
|||||||
return new PluginResult(PluginResult.Status.JSON_EXCEPTION, "Missing source or target");
|
return new PluginResult(PluginResult.Status.JSON_EXCEPTION, "Missing source or target");
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
|
||||||
if (action.equals("upload")) {
|
if (action.equals("upload")) {
|
||||||
// Setup the options
|
return upload(source, target, args);
|
||||||
String fileKey = null;
|
|
||||||
String fileName = null;
|
|
||||||
String mimeType = null;
|
|
||||||
|
|
||||||
fileKey = getArgument(args, 2, "file");
|
|
||||||
fileName = getArgument(args, 3, "image.jpg");
|
|
||||||
mimeType = getArgument(args, 4, "image/jpeg");
|
|
||||||
JSONObject params = args.optJSONObject(5);
|
|
||||||
boolean trustEveryone = args.optBoolean(6);
|
|
||||||
boolean chunkedMode = args.optBoolean(7) || args.isNull(7); //Always use chunked mode unless set to false as per API
|
|
||||||
FileUploadResult r = upload(source, target, fileKey, fileName, mimeType, params, trustEveryone, chunkedMode);
|
|
||||||
Log.d(LOG_TAG, "****** About to return a result from upload");
|
|
||||||
return new PluginResult(PluginResult.Status.OK, r.toJSONObject());
|
|
||||||
} else if (action.equals("download")) {
|
} else if (action.equals("download")) {
|
||||||
JSONObject r = download(source, target);
|
return download(source, target);
|
||||||
Log.d(LOG_TAG, "****** About to return a result from download");
|
|
||||||
return new PluginResult(PluginResult.Status.OK, r);
|
|
||||||
} else {
|
} else {
|
||||||
return new PluginResult(PluginResult.Status.INVALID_ACTION);
|
return new PluginResult(PluginResult.Status.INVALID_ACTION);
|
||||||
}
|
}
|
||||||
} catch (FileNotFoundException e) {
|
|
||||||
Log.e(LOG_TAG, e.getMessage(), e);
|
|
||||||
JSONObject error = createFileTransferError(FILE_NOT_FOUND_ERR, source, target);
|
|
||||||
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
|
||||||
} catch (IllegalArgumentException e) {
|
|
||||||
Log.e(LOG_TAG, e.getMessage(), e);
|
|
||||||
JSONObject error = createFileTransferError(INVALID_URL_ERR, source, target);
|
|
||||||
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
|
||||||
} catch (SSLException e) {
|
|
||||||
Log.e(LOG_TAG, e.getMessage(), e);
|
|
||||||
Log.d(LOG_TAG, "Got my ssl exception!!!");
|
|
||||||
JSONObject error = createFileTransferError(CONNECTION_ERR, source, target);
|
|
||||||
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
|
||||||
} catch (IOException e) {
|
|
||||||
Log.e(LOG_TAG, e.getMessage(), e);
|
|
||||||
JSONObject error = createFileTransferError(CONNECTION_ERR, source, target);
|
|
||||||
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
|
||||||
} catch (JSONException e) {
|
|
||||||
Log.e(LOG_TAG, e.getMessage(), e);
|
|
||||||
return new PluginResult(PluginResult.Status.JSON_EXCEPTION);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// always verify the host - don't check for certificate
|
|
||||||
final static HostnameVerifier DO_NOT_VERIFY = new HostnameVerifier() {
|
|
||||||
public boolean verify(String hostname, SSLSession session) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This function will install a trust manager that will blindly trust all SSL
|
* Uploads the specified file to the server URL provided using an HTTP multipart request.
|
||||||
* certificates. The reason this code is being added is to enable developers
|
* @param source Full path of the file on the file system
|
||||||
* to do development using self signed SSL certificates on their web server.
|
* @param target URL of the server to receive the file
|
||||||
|
* @param args JSON Array of args
|
||||||
*
|
*
|
||||||
* The standard HttpsURLConnection class will throw an exception on self
|
* args[2] fileKey Name of file request parameter
|
||||||
* signed certificates if this code is not run.
|
* args[3] fileName File name to be used on server
|
||||||
*/
|
* args[4] mimeType Describes file content type
|
||||||
private void trustAllHosts() {
|
* args[5] params key:value pairs of user-defined parameters
|
||||||
// Create a trust manager that does not validate certificate chains
|
|
||||||
TrustManager[] trustAllCerts = new TrustManager[] { new X509TrustManager() {
|
|
||||||
public java.security.cert.X509Certificate[] getAcceptedIssuers() {
|
|
||||||
return new java.security.cert.X509Certificate[] {};
|
|
||||||
}
|
|
||||||
|
|
||||||
public void checkClientTrusted(X509Certificate[] chain,
|
|
||||||
String authType) throws CertificateException {
|
|
||||||
}
|
|
||||||
|
|
||||||
public void checkServerTrusted(X509Certificate[] chain,
|
|
||||||
String authType) throws CertificateException {
|
|
||||||
}
|
|
||||||
} };
|
|
||||||
|
|
||||||
// Install the all-trusting trust manager
|
|
||||||
try {
|
|
||||||
// Backup the current SSL socket factory
|
|
||||||
defaultSSLSocketFactory = HttpsURLConnection.getDefaultSSLSocketFactory();
|
|
||||||
// Install our all trusting manager
|
|
||||||
SSLContext sc = SSLContext.getInstance("TLS");
|
|
||||||
sc.init(null, trustAllCerts, new java.security.SecureRandom());
|
|
||||||
HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory());
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.e(LOG_TAG, e.getMessage(), e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create an error object based on the passed in errorCode
|
|
||||||
* @param errorCode the error
|
|
||||||
* @return JSONObject containing the error
|
|
||||||
*/
|
|
||||||
private JSONObject createFileTransferError(int errorCode, String source, String target) {
|
|
||||||
JSONObject error = null;
|
|
||||||
try {
|
|
||||||
error = new JSONObject();
|
|
||||||
error.put("code", errorCode);
|
|
||||||
error.put("source", source);
|
|
||||||
error.put("target", target);
|
|
||||||
} catch (JSONException e) {
|
|
||||||
Log.e(LOG_TAG, e.getMessage(), e);
|
|
||||||
}
|
|
||||||
return error;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convenience method to read a parameter from the list of JSON args.
|
|
||||||
* @param args the args passed to the Plugin
|
|
||||||
* @param position the position to retrieve the arg from
|
|
||||||
* @param defaultString the default to be used if the arg does not exist
|
|
||||||
* @return String with the retrieved value
|
|
||||||
*/
|
|
||||||
private String getArgument(JSONArray args, int position, String defaultString) {
|
|
||||||
String arg = defaultString;
|
|
||||||
if(args.length() >= position) {
|
|
||||||
arg = args.optString(position);
|
|
||||||
if (arg == null || "null".equals(arg)) {
|
|
||||||
arg = defaultString;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return arg;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Uploads the specified file to the server URL provided using an HTTP
|
|
||||||
* multipart request.
|
|
||||||
* @param file Full path of the file on the file system
|
|
||||||
* @param server URL of the server to receive the file
|
|
||||||
* @param fileKey Name of file request parameter
|
|
||||||
* @param fileName File name to be used on server
|
|
||||||
* @param mimeType Describes file content type
|
|
||||||
* @param params key:value pairs of user-defined parameters
|
|
||||||
* @return FileUploadResult containing result of upload request
|
* @return FileUploadResult containing result of upload request
|
||||||
*/
|
*/
|
||||||
public FileUploadResult upload(String file, String server, final String fileKey, final String fileName,
|
private PluginResult upload(String source, String target, JSONArray args) {
|
||||||
final String mimeType, JSONObject params, boolean trustEveryone, boolean chunkedMode) throws IOException, SSLException {
|
Log.d(LOG_TAG, "upload " + source + " to " + target);
|
||||||
|
|
||||||
|
HttpURLConnection conn = null;
|
||||||
|
try {
|
||||||
|
// Setup the options
|
||||||
|
String fileKey = getArgument(args, 2, "file");
|
||||||
|
String fileName = getArgument(args, 3, "image.jpg");
|
||||||
|
String mimeType = getArgument(args, 4, "image/jpeg");
|
||||||
|
JSONObject params = args.optJSONObject(5);
|
||||||
|
if (params == null) params = new JSONObject();
|
||||||
|
boolean trustEveryone = args.optBoolean(6);
|
||||||
|
boolean chunkedMode = args.optBoolean(7) || args.isNull(7); //Always use chunked mode unless set to false as per API
|
||||||
|
|
||||||
|
Log.d(LOG_TAG, "fileKey: " + fileKey);
|
||||||
|
Log.d(LOG_TAG, "fileName: " + fileName);
|
||||||
|
Log.d(LOG_TAG, "mimeType: " + mimeType);
|
||||||
|
Log.d(LOG_TAG, "params: " + params);
|
||||||
|
Log.d(LOG_TAG, "trustEveryone: " + trustEveryone);
|
||||||
|
Log.d(LOG_TAG, "chunkedMode: " + chunkedMode);
|
||||||
|
|
||||||
// Create return object
|
// Create return object
|
||||||
FileUploadResult result = new FileUploadResult();
|
FileUploadResult result = new FileUploadResult();
|
||||||
|
|
||||||
// Get a input stream of the file on the phone
|
// Get a input stream of the file on the phone
|
||||||
FileInputStream fileInputStream = (FileInputStream) getPathFromUri(file);
|
FileInputStream fileInputStream = (FileInputStream) getPathFromUri(source);
|
||||||
|
|
||||||
HttpURLConnection conn = null;
|
|
||||||
DataOutputStream dos = null;
|
DataOutputStream dos = null;
|
||||||
|
|
||||||
int bytesRead, bytesAvailable, bufferSize;
|
int bytesRead, bytesAvailable, bufferSize;
|
||||||
@ -237,7 +139,7 @@ public class FileTransfer extends Plugin {
|
|||||||
|
|
||||||
//------------------ CLIENT REQUEST
|
//------------------ CLIENT REQUEST
|
||||||
// open a URL connection to the server
|
// open a URL connection to the server
|
||||||
URL url = new URL(server);
|
URL url = new URL(target);
|
||||||
|
|
||||||
// Open a HTTP connection to the URL based on protocol
|
// Open a HTTP connection to the URL based on protocol
|
||||||
if (url.getProtocol().toLowerCase().equals("https")) {
|
if (url.getProtocol().toLowerCase().equals("https")) {
|
||||||
@ -275,7 +177,7 @@ public class FileTransfer extends Plugin {
|
|||||||
// Use a post method.
|
// Use a post method.
|
||||||
conn.setRequestMethod("POST");
|
conn.setRequestMethod("POST");
|
||||||
conn.setRequestProperty("Connection", "Keep-Alive");
|
conn.setRequestProperty("Connection", "Keep-Alive");
|
||||||
conn.setRequestProperty("Content-Type", "multipart/form-data;boundary="+BOUNDRY);
|
conn.setRequestProperty("Content-Type", "multipart/form-data;boundary=" + BOUNDARY);
|
||||||
|
|
||||||
// Handle the other headers
|
// Handle the other headers
|
||||||
try {
|
try {
|
||||||
@ -290,7 +192,7 @@ public class FileTransfer extends Plugin {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Set the cookies on the response
|
// Set the cookies on the response
|
||||||
String cookie = CookieManager.getInstance().getCookie(server);
|
String cookie = CookieManager.getInstance().getCookie(target);
|
||||||
if (cookie != null) {
|
if (cookie != null) {
|
||||||
conn.setRequestProperty("Cookie", cookie);
|
conn.setRequestProperty("Cookie", cookie);
|
||||||
}
|
}
|
||||||
@ -304,9 +206,9 @@ public class FileTransfer extends Plugin {
|
|||||||
try {
|
try {
|
||||||
for (Iterator iter = params.keys(); iter.hasNext();) {
|
for (Iterator iter = params.keys(); iter.hasNext();) {
|
||||||
Object key = iter.next();
|
Object key = iter.next();
|
||||||
if(key.toString() != "headers")
|
if(!String.valueOf(key).equals("headers"))
|
||||||
{
|
{
|
||||||
extraParams += LINE_START + BOUNDRY + LINE_END;
|
extraParams += LINE_START + BOUNDARY + LINE_END;
|
||||||
extraParams += "Content-Disposition: form-data; name=\"" + key.toString() + "\";";
|
extraParams += "Content-Disposition: form-data; name=\"" + key.toString() + "\";";
|
||||||
extraParams += LINE_END + LINE_END;
|
extraParams += LINE_END + LINE_END;
|
||||||
extraParams += params.getString(key.toString());
|
extraParams += params.getString(key.toString());
|
||||||
@ -317,11 +219,11 @@ public class FileTransfer extends Plugin {
|
|||||||
Log.e(LOG_TAG, e.getMessage(), e);
|
Log.e(LOG_TAG, e.getMessage(), e);
|
||||||
}
|
}
|
||||||
|
|
||||||
extraParams += LINE_START + BOUNDRY + LINE_END;
|
extraParams += LINE_START + BOUNDARY + LINE_END;
|
||||||
extraParams += "Content-Disposition: form-data; name=\"" + fileKey + "\";" + " filename=\"";
|
extraParams += "Content-Disposition: form-data; name=\"" + fileKey + "\";" + " filename=\"";
|
||||||
|
|
||||||
String midParams = "\"" + LINE_END + "Content-Type: " + mimeType + LINE_END + LINE_END;
|
String midParams = "\"" + LINE_END + "Content-Type: " + mimeType + LINE_END + LINE_END;
|
||||||
String tailParams = LINE_END + LINE_START + BOUNDRY + LINE_START + LINE_END;
|
String tailParams = LINE_END + LINE_START + BOUNDARY + LINE_START + LINE_END;
|
||||||
|
|
||||||
// Should set this up as an option
|
// Should set this up as an option
|
||||||
if (chunkedMode) {
|
if (chunkedMode) {
|
||||||
@ -375,6 +277,7 @@ public class FileTransfer extends Plugin {
|
|||||||
try {
|
try {
|
||||||
inStream = new DataInputStream ( conn.getInputStream() );
|
inStream = new DataInputStream ( conn.getInputStream() );
|
||||||
} catch(FileNotFoundException e) {
|
} catch(FileNotFoundException e) {
|
||||||
|
Log.e(LOG_TAG, e.toString(), e);
|
||||||
throw new IOException("Received error from server");
|
throw new IOException("Received error from server");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -390,7 +293,6 @@ public class FileTransfer extends Plugin {
|
|||||||
result.setResponse(responseString.toString());
|
result.setResponse(responseString.toString());
|
||||||
|
|
||||||
inStream.close();
|
inStream.close();
|
||||||
conn.disconnect();
|
|
||||||
|
|
||||||
// Revert back to the proper verifier and socket factories
|
// Revert back to the proper verifier and socket factories
|
||||||
if (trustEveryone && url.getProtocol().toLowerCase().equals("https")) {
|
if (trustEveryone && url.getProtocol().toLowerCase().equals("https")) {
|
||||||
@ -398,8 +300,134 @@ public class FileTransfer extends Plugin {
|
|||||||
HttpsURLConnection.setDefaultSSLSocketFactory(defaultSSLSocketFactory);
|
HttpsURLConnection.setDefaultSSLSocketFactory(defaultSSLSocketFactory);
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
Log.d(LOG_TAG, "****** About to return a result from upload");
|
||||||
|
return new PluginResult(PluginResult.Status.OK, result.toJSONObject());
|
||||||
|
|
||||||
|
} catch (FileNotFoundException e) {
|
||||||
|
JSONObject error = createFileTransferError(FILE_NOT_FOUND_ERR, source, target, conn);
|
||||||
|
Log.e(LOG_TAG, error.toString(), e);
|
||||||
|
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
||||||
|
} catch (MalformedURLException e) {
|
||||||
|
JSONObject error = createFileTransferError(INVALID_URL_ERR, source, target, conn);
|
||||||
|
Log.e(LOG_TAG, error.toString(), e);
|
||||||
|
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
||||||
|
} catch (IOException e) {
|
||||||
|
JSONObject error = createFileTransferError(CONNECTION_ERR, source, target, conn);
|
||||||
|
Log.e(LOG_TAG, error.toString(), e);
|
||||||
|
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
||||||
|
} catch (JSONException e) {
|
||||||
|
Log.e(LOG_TAG, e.getMessage(), e);
|
||||||
|
return new PluginResult(PluginResult.Status.JSON_EXCEPTION);
|
||||||
|
} catch (Throwable t) {
|
||||||
|
// Shouldn't happen, but will
|
||||||
|
JSONObject error = createFileTransferError(CONNECTION_ERR, source, target, conn);
|
||||||
|
Log.wtf(LOG_TAG, error.toString(), t);
|
||||||
|
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
||||||
|
} finally {
|
||||||
|
if (conn != null) {
|
||||||
|
conn.disconnect();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// always verify the host - don't check for certificate
|
||||||
|
final static HostnameVerifier DO_NOT_VERIFY = new HostnameVerifier() {
|
||||||
|
public boolean verify(String hostname, SSLSession session) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This function will install a trust manager that will blindly trust all SSL
|
||||||
|
* certificates. The reason this code is being added is to enable developers
|
||||||
|
* to do development using self signed SSL certificates on their web server.
|
||||||
|
*
|
||||||
|
* The standard HttpsURLConnection class will throw an exception on self
|
||||||
|
* signed certificates if this code is not run.
|
||||||
|
*/
|
||||||
|
private void trustAllHosts() {
|
||||||
|
// Create a trust manager that does not validate certificate chains
|
||||||
|
TrustManager[] trustAllCerts = new TrustManager[] { new X509TrustManager() {
|
||||||
|
public java.security.cert.X509Certificate[] getAcceptedIssuers() {
|
||||||
|
return new java.security.cert.X509Certificate[] {};
|
||||||
|
}
|
||||||
|
|
||||||
|
public void checkClientTrusted(X509Certificate[] chain,
|
||||||
|
String authType) throws CertificateException {
|
||||||
|
}
|
||||||
|
|
||||||
|
public void checkServerTrusted(X509Certificate[] chain,
|
||||||
|
String authType) throws CertificateException {
|
||||||
|
}
|
||||||
|
} };
|
||||||
|
|
||||||
|
// Install the all-trusting trust manager
|
||||||
|
try {
|
||||||
|
// Backup the current SSL socket factory
|
||||||
|
defaultSSLSocketFactory = HttpsURLConnection.getDefaultSSLSocketFactory();
|
||||||
|
// Install our all trusting manager
|
||||||
|
SSLContext sc = SSLContext.getInstance("TLS");
|
||||||
|
sc.init(null, trustAllCerts, new java.security.SecureRandom());
|
||||||
|
HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory());
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.e(LOG_TAG, e.getMessage(), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private JSONObject createFileTransferError(int errorCode, String source, String target, HttpURLConnection connection) {
|
||||||
|
|
||||||
|
Integer httpStatus = null;
|
||||||
|
|
||||||
|
if (connection != null) {
|
||||||
|
try {
|
||||||
|
httpStatus = connection.getResponseCode();
|
||||||
|
} catch (IOException e) {
|
||||||
|
Log.w(LOG_TAG, "Error getting HTTP status code from connection.", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return createFileTransferError(errorCode, source, target, httpStatus);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an error object based on the passed in errorCode
|
||||||
|
* @param errorCode the error
|
||||||
|
* @return JSONObject containing the error
|
||||||
|
*/
|
||||||
|
private JSONObject createFileTransferError(int errorCode, String source, String target, Integer httpStatus) {
|
||||||
|
JSONObject error = null;
|
||||||
|
try {
|
||||||
|
error = new JSONObject();
|
||||||
|
error.put("code", errorCode);
|
||||||
|
error.put("source", source);
|
||||||
|
error.put("target", target);
|
||||||
|
if (httpStatus != null) {
|
||||||
|
error.put("http_status", httpStatus);
|
||||||
|
}
|
||||||
|
} catch (JSONException e) {
|
||||||
|
Log.e(LOG_TAG, e.getMessage(), e);
|
||||||
|
}
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenience method to read a parameter from the list of JSON args.
|
||||||
|
* @param args the args passed to the Plugin
|
||||||
|
* @param position the position to retrieve the arg from
|
||||||
|
* @param defaultString the default to be used if the arg does not exist
|
||||||
|
* @return String with the retrieved value
|
||||||
|
*/
|
||||||
|
private String getArgument(JSONArray args, int position, String defaultString) {
|
||||||
|
String arg = defaultString;
|
||||||
|
if(args.length() >= position) {
|
||||||
|
arg = args.optString(position);
|
||||||
|
if (arg == null || "null".equals(arg)) {
|
||||||
|
arg = defaultString;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return arg;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Downloads a file form a given URL and saves it to the specified directory.
|
* Downloads a file form a given URL and saves it to the specified directory.
|
||||||
@ -408,7 +436,10 @@ public class FileTransfer extends Plugin {
|
|||||||
* @param target Full path of the file on the file system
|
* @param target Full path of the file on the file system
|
||||||
* @return JSONObject the downloaded file
|
* @return JSONObject the downloaded file
|
||||||
*/
|
*/
|
||||||
public JSONObject download(String source, String target) throws IOException {
|
private PluginResult download(String source, String target) {
|
||||||
|
Log.d(LOG_TAG, "download " + source + " to " + target);
|
||||||
|
|
||||||
|
HttpURLConnection connection = null;
|
||||||
try {
|
try {
|
||||||
File file = getFileFromPath(target);
|
File file = getFileFromPath(target);
|
||||||
|
|
||||||
@ -419,7 +450,7 @@ public class FileTransfer extends Plugin {
|
|||||||
if(this.ctx.isUrlWhiteListed(source))
|
if(this.ctx.isUrlWhiteListed(source))
|
||||||
{
|
{
|
||||||
URL url = new URL(source);
|
URL url = new URL(source);
|
||||||
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
|
connection = (HttpURLConnection) url.openConnection();
|
||||||
connection.setRequestMethod("GET");
|
connection.setRequestMethod("GET");
|
||||||
|
|
||||||
//Add cookie support
|
//Add cookie support
|
||||||
@ -450,23 +481,40 @@ public class FileTransfer extends Plugin {
|
|||||||
|
|
||||||
// create FileEntry object
|
// create FileEntry object
|
||||||
FileUtils fileUtil = new FileUtils();
|
FileUtils fileUtil = new FileUtils();
|
||||||
|
JSONObject fileEntry = fileUtil.getEntry(file);
|
||||||
|
|
||||||
return fileUtil.getEntry(file);
|
return new PluginResult(PluginResult.Status.OK, fileEntry);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
throw new IOException("Error: Unable to connect to domain");
|
Log.w(LOG_TAG, "Source URL is not in white list: '" + source + "'");
|
||||||
|
JSONObject error = createFileTransferError(CONNECTION_ERR, source, target, 401);
|
||||||
|
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (FileNotFoundException e) {
|
||||||
|
JSONObject error = createFileTransferError(FILE_NOT_FOUND_ERR, source, target, connection);
|
||||||
|
Log.e(LOG_TAG, error.toString(), e);
|
||||||
|
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
||||||
|
} catch (MalformedURLException e) {
|
||||||
|
JSONObject error = createFileTransferError(INVALID_URL_ERR, source, target, connection);
|
||||||
|
Log.e(LOG_TAG, error.toString(), e);
|
||||||
|
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
||||||
|
} catch (Exception e) { // IOException, JSONException, NullPointer
|
||||||
|
JSONObject error = createFileTransferError(CONNECTION_ERR, source, target, connection);
|
||||||
|
Log.e(LOG_TAG, error.toString(), e);
|
||||||
|
return new PluginResult(PluginResult.Status.IO_EXCEPTION, error);
|
||||||
|
} finally {
|
||||||
|
if (connection != null) {
|
||||||
|
connection.disconnect();
|
||||||
}
|
}
|
||||||
} catch (Exception e) {
|
|
||||||
Log.d(LOG_TAG, e.getMessage(), e);
|
|
||||||
throw new IOException("Error while downloading");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get an input stream based on file path or content:// uri
|
* Get an input stream based on file path or content:// uri
|
||||||
*
|
*
|
||||||
* @param path
|
* @param path foo
|
||||||
* @return an input stream
|
* @return an input stream
|
||||||
* @throws FileNotFoundException
|
* @throws FileNotFoundException
|
||||||
*/
|
*/
|
||||||
@ -491,14 +539,23 @@ public class FileTransfer extends Plugin {
|
|||||||
/**
|
/**
|
||||||
* Get a File object from the passed in path
|
* Get a File object from the passed in path
|
||||||
*
|
*
|
||||||
* @param path
|
* @param path file path
|
||||||
* @return
|
* @return file object
|
||||||
*/
|
*/
|
||||||
private File getFileFromPath(String path) {
|
private File getFileFromPath(String path) throws FileNotFoundException {
|
||||||
if (path.startsWith("file://")) {
|
File file;
|
||||||
return new File(path.substring(7));
|
String prefix = "file://";
|
||||||
|
|
||||||
|
if (path.startsWith(prefix)) {
|
||||||
|
file = new File(path.substring(prefix.length()));
|
||||||
} else {
|
} else {
|
||||||
return new File(path);
|
file = new File(path);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (file.getParent() == null) {
|
||||||
|
throw new FileNotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
return file;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user