diff --git a/packages/background-http/index.ios.ts b/packages/background-http/index.ios.ts index 04876e6e..d468b88f 100644 --- a/packages/background-http/index.ios.ts +++ b/packages/background-http/index.ios.ts @@ -1,410 +1,402 @@ -import { Observable, knownFolders } from "@nativescript/core"; -import { ProgressEventData, ErrorEventData, ResultEventData, Session as ISession, Task as ITask, Request as IRequest, CompleteEventData } from "."; +import { Observable, knownFolders } from '@nativescript/core'; +import { ProgressEventData, ErrorEventData, ResultEventData, Session as ISession, Task as ITask, Request as IRequest, CompleteEventData } from '.'; const main_queue = dispatch_get_current_queue(); let zonedOnProgress = null; let zonedOnError = null; function onProgress(nsSession, nsTask, sent, expectedTotal) { - const task = Task.getTask(nsSession, nsTask); - task.notifyPropertyChange("upload", task.upload); - task.notifyPropertyChange("totalUpload", task.totalUpload); - task.notify({ - eventName: "progress", - object: task, - currentBytes: sent, - totalBytes: expectedTotal - }); + const task = Task.getTask(nsSession, nsTask); + task.notifyPropertyChange('upload', task.upload); + task.notifyPropertyChange('totalUpload', task.totalUpload); + task.notify({ + eventName: 'progress', + object: task, + currentBytes: sent, + totalBytes: expectedTotal, + }); } function onError(session, nsTask, error) { - const task = Task.getTask(session, nsTask); - if (task._fileToCleanup) { - NSFileManager.defaultManager.removeItemAtPathError(task._fileToCleanup); - } - const response = nsTask && nsTask.performSelector("response"); - if (error) { - task.notifyPropertyChange("status", task.status); - task.notify({ - eventName: "error", - object: task, - error, - responseCode: response ? response.statusCode : -1, - response - }); - } else { - task.notifyPropertyChange("upload", task.upload); - task.notifyPropertyChange("totalUpload", task.totalUpload); - task.notify({ - eventName: "progress", - object: task, - currentBytes: nsTask.countOfBytesSent, - totalBytes: nsTask.countOfBytesExpectedToSend - }); - task.notify({ - eventName: "complete", - object: task, - responseCode: response ? response.statusCode : -1, - response - }); - - Task._tasks.delete(nsTask); - } + const task = Task.getTask(session, nsTask); + if (task._fileToCleanup) { + NSFileManager.defaultManager.removeItemAtPathError(task._fileToCleanup); + } + const response = nsTask && nsTask.performSelector('response'); + if (error) { + task.notifyPropertyChange('status', task.status); + task.notify({ + eventName: 'error', + object: task, + error, + responseCode: response ? response.statusCode : -1, + response, + }); + } else { + task.notifyPropertyChange('upload', task.upload); + task.notifyPropertyChange('totalUpload', task.totalUpload); + task.notify({ + eventName: 'progress', + object: task, + currentBytes: nsTask.countOfBytesSent, + totalBytes: nsTask.countOfBytesExpectedToSend, + }); + task.notify({ + eventName: 'complete', + object: task, + responseCode: response ? response.statusCode : -1, + response, + }); + + Task._tasks.delete(nsTask); + } } @NativeClass() class BackgroundUploadDelegate extends NSObject implements NSURLSessionDelegate, NSURLSessionTaskDelegate, NSURLSessionDataDelegate, NSURLSessionDownloadDelegate { - - static ObjCProtocols = [NSURLSessionDelegate, NSURLSessionTaskDelegate, NSURLSessionDataDelegate, NSURLSessionDownloadDelegate]; - - // NSURLSessionDelegate - URLSessionDidBecomeInvalidWithError(session, error) { - } - - URLSessionDidReceiveChallengeCompletionHandler(session, challenge, comlpetionHandler) { - const disposition = null; - const credential = null; - comlpetionHandler(disposition, credential); - } - - URLSessionDidFinishEventsForBackgroundURLSession(session) { - } - - // NSURLSessionTaskDelegate - URLSessionTaskDidCompleteWithError(session: NSURLSession, nsTask: NSURLSessionTask, error: NSError) { - dispatch_async(main_queue, () => { - zonedOnError(session, nsTask, error); - }); - } - - URLSessionTaskDidReceiveChallengeCompletionHandler(session, task, challenge, completionHandler) { - const disposition = null; - const credential = null; - completionHandler(disposition, credential); - } - - URLSessionTaskDidSendBodyDataTotalBytesSentTotalBytesExpectedToSend(nsSession: NSURLSession, nsTask: NSURLSessionTask, data, sent: number, expectedTotal: number) { - dispatch_async(main_queue, () => { - zonedOnProgress(nsSession, nsTask, sent, expectedTotal); - }); - } - - URLSessionTaskNeedNewBodyStream(session, task, need) { - } - - URLSessionTaskWillPerformHTTPRedirectionNewRequestCompletionHandler(session, task, redirect, request, completionHandler) { - completionHandler(request); - } - - // NSURLSessionDataDelegate - URLSessionDataTaskDidReceiveResponseCompletionHandler(session, dataTask, response, completionHandler) { - const disposition = null; - completionHandler(disposition); - } - - URLSessionDataTaskDidBecomeDownloadTask(session, dataTask, downloadTask) { - } - - URLSessionDataTaskDidReceiveData(session: NSURLSession, dataTask: NSURLSessionDataTask, data: NSData) { - dispatch_async(main_queue, () => { - // we have a response in the data... - const jsTask = Task.getTask(session, dataTask); - const jsonString = NSString.alloc().initWithDataEncoding(data, NSUTF8StringEncoding); - - jsTask.notify({ - eventName: "responded", - object: jsTask, - data: jsonString.toString(), - responseCode: dataTask && dataTask.response ? (dataTask.response).statusCode : -1 - }); - }); - } - - URLSessionDataTaskWillCacheResponseCompletionHandler() { - } - - // NSURLSessionDownloadDelegate - URLSessionDownloadTaskDidResumeAtOffsetExpectedTotalBytes(session, task, offset, expects) { - } - - URLSessionDownloadTaskDidWriteDataTotalBytesWrittenTotalBytesExpectedToWrite(session, task, data, written, expected) { - } - - URLSessionDownloadTaskDidFinishDownloadingToURL(session, task, url) { - } + static ObjCProtocols = [NSURLSessionDelegate, NSURLSessionTaskDelegate, NSURLSessionDataDelegate, NSURLSessionDownloadDelegate]; + + // NSURLSessionDelegate + URLSessionDidBecomeInvalidWithError(session, error) {} + + URLSessionDidReceiveChallengeCompletionHandler(session, challenge, comlpetionHandler) { + const disposition = null; + const credential = null; + comlpetionHandler(disposition, credential); + } + + URLSessionDidFinishEventsForBackgroundURLSession(session) {} + + // NSURLSessionTaskDelegate + URLSessionTaskDidCompleteWithError(session: NSURLSession, nsTask: NSURLSessionTask, error: NSError) { + dispatch_async(main_queue, () => { + zonedOnError(session, nsTask, error); + }); + } + + URLSessionTaskDidReceiveChallengeCompletionHandler(session, task, challenge, completionHandler) { + const disposition = null; + const credential = null; + completionHandler(disposition, credential); + } + + URLSessionTaskDidSendBodyDataTotalBytesSentTotalBytesExpectedToSend(nsSession: NSURLSession, nsTask: NSURLSessionTask, data, sent: number, expectedTotal: number) { + dispatch_async(main_queue, () => { + zonedOnProgress(nsSession, nsTask, sent, expectedTotal); + }); + } + + URLSessionTaskNeedNewBodyStream(session, task, need) {} + + URLSessionTaskWillPerformHTTPRedirectionNewRequestCompletionHandler(session, task, redirect, request, completionHandler) { + completionHandler(request); + } + + // NSURLSessionDataDelegate + URLSessionDataTaskDidReceiveResponseCompletionHandler(session, dataTask, response, completionHandler) { + const disposition = null; + completionHandler(disposition); + } + + URLSessionDataTaskDidBecomeDownloadTask(session, dataTask, downloadTask) {} + + URLSessionDataTaskDidReceiveData(session: NSURLSession, dataTask: NSURLSessionDataTask, data: NSData) { + dispatch_async(main_queue, () => { + // we have a response in the data... + const jsTask = Task.getTask(session, dataTask); + const jsonString = NSString.alloc().initWithDataEncoding(data, NSUTF8StringEncoding); + + jsTask.notify({ + eventName: 'responded', + object: jsTask, + data: jsonString.toString(), + responseCode: dataTask && dataTask.response ? (dataTask.response).statusCode : -1, + }); + }); + } + + URLSessionDataTaskWillCacheResponseCompletionHandler() {} + + // NSURLSessionDownloadDelegate + URLSessionDownloadTaskDidResumeAtOffsetExpectedTotalBytes(session, task, offset, expects) {} + + URLSessionDownloadTaskDidWriteDataTotalBytesWrittenTotalBytesExpectedToWrite(session, task, data, written, expected) {} + + URLSessionDownloadTaskDidFinishDownloadingToURL(session, task, url) {} } class Session implements Session { - // TODO: Create a mechanism to clean sessions from the cache that have all their tasks completed, canceled or errored out. - private static _sessions: { [id: string]: Session } = {}; - - private _session: NSURLSession; - - constructor(id: string) { - const delegate = BackgroundUploadDelegate.alloc().init(); - const configuration = NSURLSessionConfiguration.backgroundSessionConfigurationWithIdentifier(id); - this._session = NSURLSession.sessionWithConfigurationDelegateDelegateQueue(configuration, delegate, null); - zonedOnProgress = global.zonedCallback(onProgress); - zonedOnError = global.zonedCallback(onError); - } - - get ios(): any { - return this._session; - } - - - public uploadFile(fileUri: string, options: IRequest): Task { - if (!fileUri) { - throw new Error("File must be provided."); - } - - const url = NSURL.URLWithString(options.url); - const request = NSMutableURLRequest.requestWithURL(url); - - const headers = options.headers; - if (headers) { - for (let header in headers) { - const value = headers[header]; - if (value !== null && value !== void 0) { - request.setValueForHTTPHeaderField(value.toString(), header); - } - } - } - - if (options.method) { - request.HTTPMethod = options.method; - } - - let fileURL: NSURL; - if (fileUri.substr(0, 7) === "file://") { - // File URI in string format - fileURL = NSURL.URLWithString(fileUri); - } else if (fileUri.charAt(0) === "/") { - // Absolute path with leading slash - fileURL = NSURL.fileURLWithPath(fileUri); - } - - const newTask = this._session.uploadTaskWithRequestFromFile(request, fileURL); - newTask.taskDescription = options.description; - newTask.resume(); - const retTask: Task = Task.getTask(this._session, newTask); - return retTask; - } - public multipartUpload(params: any[], options: any): Task { - const MPF = new MultiMultiPartForm(); - for (let i = 0; i < params.length; i++) { - const curParam = params[i]; - if (typeof curParam.name === 'undefined') { - throw new Error("You must have a `name` value"); - } - - if (curParam.filename) { - const destFileName = curParam.destFilename || curParam.filename.substring(curParam.filename.lastIndexOf('/') + 1, curParam.filename.length); - MPF.appendParam(curParam.name, null, curParam.filename, curParam.mimeType, destFileName); - } else { - MPF.appendParam(curParam.name, curParam.value); - } - } - const header = MPF.getHeader(); - const uploadFile = MPF.generateFile(); - - if (!options.headers) { - options.headers = {}; - } - options.headers['Content-Type'] = header['Content-Type']; - - const task = this.uploadFile(uploadFile, options); - - // Tag the file to be deleted and cleanup after upload - (task)._fileToCleanup = uploadFile; - return task; - } - static getSession(id: string): Session { - let jsSession = Session._sessions[id]; - if (jsSession) { - return jsSession; - } - jsSession = new Session(id); - Session._sessions[id] = jsSession; - return jsSession; - } + // TODO: Create a mechanism to clean sessions from the cache that have all their tasks completed, canceled or errored out. + private static _sessions: { [id: string]: Session } = {}; + + private _session: NSURLSession; + + constructor(id: string) { + const delegate = BackgroundUploadDelegate.alloc().init(); + const configuration = NSURLSessionConfiguration.backgroundSessionConfigurationWithIdentifier(id); + this._session = NSURLSession.sessionWithConfigurationDelegateDelegateQueue(configuration, delegate, null); + zonedOnProgress = global.zonedCallback(onProgress); + zonedOnError = global.zonedCallback(onError); + } + + get ios(): any { + return this._session; + } + + public uploadFile(fileUri: string, options: IRequest): Task { + if (!fileUri) { + throw new Error('File must be provided.'); + } + + const url = NSURL.URLWithString(options.url); + const request = NSMutableURLRequest.requestWithURL(url); + + const headers = options.headers; + if (headers) { + for (let header in headers) { + const value = headers[header]; + if (value !== null && value !== void 0) { + request.setValueForHTTPHeaderField(value.toString(), header); + } + } + } + + if (options.method) { + request.HTTPMethod = options.method; + } + + let fileURL: NSURL; + if (fileUri.substr(0, 7) === 'file://') { + // File URI in string format + fileURL = NSURL.URLWithString(fileUri); + } else if (fileUri.charAt(0) === '/') { + // Absolute path with leading slash + fileURL = NSURL.fileURLWithPath(fileUri); + } + + const newTask = this._session.uploadTaskWithRequestFromFile(request, fileURL); + newTask.taskDescription = options.description; + newTask.resume(); + const retTask: Task = Task.getTask(this._session, newTask); + return retTask; + } + public multipartUpload(params: any[], options: any): Task { + const MPF = new MultiMultiPartForm(); + for (let i = 0; i < params.length; i++) { + const curParam = params[i]; + if (typeof curParam.name === 'undefined') { + throw new Error('You must have a `name` value'); + } + + if (curParam.filename) { + const destFileName = curParam.destFilename || curParam.filename.substring(curParam.filename.lastIndexOf('/') + 1, curParam.filename.length); + MPF.appendParam(curParam.name, null, curParam.filename, curParam.mimeType, destFileName); + } else { + MPF.appendParam(curParam.name, curParam.value); + } + } + const header = MPF.getHeader(); + const uploadFile = MPF.generateFile(); + + if (!options.headers) { + options.headers = {}; + } + options.headers['Content-Type'] = header['Content-Type']; + + const task = this.uploadFile(uploadFile, options); + + // Tag the file to be deleted and cleanup after upload + (task)._fileToCleanup = uploadFile; + return task; + } + static getSession(id: string): Session { + let jsSession = Session._sessions[id]; + if (jsSession) { + return jsSession; + } + jsSession = new Session(id); + Session._sessions[id] = jsSession; + return jsSession; + } } class NativePropertyReader { - private _invocationCache = new Map(); + private _invocationCache = new Map(); - private getInvocationObject(object: NSObject, selector: string): NSInvocation { - let invocation = this._invocationCache.get(selector); - if (!invocation) { - const sig = object.methodSignatureForSelector(selector); - invocation = NSInvocation.invocationWithMethodSignature(sig); - invocation.selector = selector; + private getInvocationObject(object: NSObject, selector: string): NSInvocation { + let invocation = this._invocationCache.get(selector); + if (!invocation) { + const sig = object.methodSignatureForSelector(selector); + invocation = NSInvocation.invocationWithMethodSignature(sig); + invocation.selector = selector; - this._invocationCache[selector] = invocation; - } + this._invocationCache[selector] = invocation; + } - return invocation; - } + return invocation; + } - public readProp(object: NSObject, prop: string, type: interop.Type): T { - const invocation = this.getInvocationObject(object, prop); - invocation.invokeWithTarget(object); + public readProp(object: NSObject, prop: string, type: interop.Type): T { + const invocation = this.getInvocationObject(object, prop); + invocation.invokeWithTarget(object); - const ret = new interop.Reference(type, new interop.Pointer()); - invocation.getReturnValue(ret); + const ret = new interop.Reference(type, new interop.Pointer()); + //invocation.getReturnValue(ret); - return ret.value; - } + return ret.value; + } } class Task extends Observable { - public static _tasks = new Map(); - public static tasksReader = new NativePropertyReader(); - private static is64BitArchitecture = interop.sizeof(interop.types.id) === 8; - public static NSIntegerType = Task.is64BitArchitecture ? interop.types.int64 : interop.types.int32; - - public _fileToCleanup: string; - private _task: NSURLSessionTask; - private _session: NSURLSession; - - constructor(nsSession: NSURLSession, nsTask: NSURLSessionTask) { - super(); - this._task = nsTask; - this._session = nsSession; - } - - get ios(): any { - return this._task; - } - - get description(): string { - return this._task.taskDescription; - } - - get upload(): number { - return Task.tasksReader.readProp(this._task, "countOfBytesSent", interop.types.int64); - } - - get totalUpload(): number { - return Task.tasksReader.readProp(this._task, "countOfBytesExpectedToSend", interop.types.int64); - } - - get status(): string { - if (Task.tasksReader.readProp(this._task, "error", Task.NSIntegerType)) { - return "error"; - } - // NSURLSessionTaskState : NSInteger, so we should pass number format here - switch (Task.tasksReader.readProp(this._task, "state", Task.NSIntegerType) as NSURLSessionTaskState) { - case NSURLSessionTaskState.Running: return "uploading"; - case NSURLSessionTaskState.Completed: return "complete"; - case NSURLSessionTaskState.Canceling: return "error"; - case NSURLSessionTaskState.Suspended: return "pending"; - } - } - - public static getTask(nsSession: NSURLSession, nsTask: NSURLSessionTask): Task { - let task = Task._tasks.get(nsTask); - if (task) { - return task; - } - - task = new Task(nsSession, nsTask); - Task._tasks.set(nsTask, task); - - return task; - } - - public cancel(): void { - this._task.cancel(); - } + public static _tasks = new Map(); + public static tasksReader = new NativePropertyReader(); + private static is64BitArchitecture = interop.sizeof(interop.types.id) === 8; + public static NSIntegerType = Task.is64BitArchitecture ? interop.types.int64 : interop.types.int32; + + public _fileToCleanup: string; + private _task: NSURLSessionTask; + private _session: NSURLSession; + + constructor(nsSession: NSURLSession, nsTask: NSURLSessionTask) { + super(); + this._task = nsTask; + this._session = nsSession; + } + + get ios(): any { + return this._task; + } + + get description(): string { + return this._task.taskDescription; + } + + get upload(): number { + return Task.tasksReader.readProp(this._task, 'countOfBytesSent', interop.types.int64); + } + + get totalUpload(): number { + return Task.tasksReader.readProp(this._task, 'countOfBytesExpectedToSend', interop.types.int64); + } + + get status(): string { + if (Task.tasksReader.readProp(this._task, 'error', Task.NSIntegerType)) { + return 'error'; + } + // NSURLSessionTaskState : NSInteger, so we should pass number format here + switch (Task.tasksReader.readProp(this._task, 'state', Task.NSIntegerType) as NSURLSessionTaskState) { + case NSURLSessionTaskState.Running: + return 'uploading'; + case NSURLSessionTaskState.Completed: + return 'complete'; + case NSURLSessionTaskState.Canceling: + return 'error'; + case NSURLSessionTaskState.Suspended: + return 'pending'; + } + } + + public static getTask(nsSession: NSURLSession, nsTask: NSURLSessionTask): Task { + let task = Task._tasks.get(nsTask); + if (task) { + return task; + } + + task = new Task(nsSession, nsTask); + Task._tasks.set(nsTask, task); + + return task; + } + + public cancel(): void { + this._task.cancel(); + } } export function session(id: string): Session { - return Session.getSession(id); + return Session.getSession(id); } class MultiMultiPartForm { - private boundary: string; - private header: any; - private fileCount: number; - private fields: Array; - - constructor() { - this.clear(); - } - - public clear(): void { - this.boundary = "--------------formboundary" + Math.floor(Math.random() * 100000000000); - this.header = { "Content-Type": 'multipart/form-data; boundary=' + this.boundary }; - this.fileCount = 0; - this.fields = []; - } - - public appendParam(name: string, value: string, filename?: string, mimeType?: string, destFileName?: string): void { - // If all we are doing is passing a field, we just add it to the fields list - if (filename == null) { - this.fields.push({ name: name, value: value }); - return; - } - // Load file - mimeType = mimeType || "application/data"; - - if (filename.startsWith("~/")) { - filename = filename.replace("~/", knownFolders.currentApp().path + "/"); - } - - const finalName = destFileName || filename.substr(filename.lastIndexOf('/') + 1, filename.length); - this.fields.push({ name: name, filename: filename, destFilename: finalName, mimeType: mimeType }); - } - - public generateFile(): string { - const CRLF = "\r\n"; - - const fileName = knownFolders.documents().path + "/temp-MPF-" + Math.floor(Math.random() * 100000000000) + ".tmp"; - - const combinedData = NSMutableData.alloc().init(); - - let results: string = ""; - let tempString: NSString; - let newData: any; - for (let i = 0; i < this.fields.length; i++) { - results += "--" + this.boundary + CRLF; - results += 'Content-Disposition: form-data; name="' + this.fields[i].name + '"'; - if (!this.fields[i].filename) { - results += CRLF + CRLF + this.fields[i].value + CRLF; - } else { - results += '; filename="' + this.fields[i].destFilename + '"'; - if (this.fields[i].mimeType) { - results += CRLF + "Content-Type: " + this.fields[i].mimeType; - } - results += CRLF + CRLF; - } - - tempString = NSString.stringWithString(results); - results = ""; - newData = tempString.dataUsingEncoding(NSUTF8StringEncoding); - combinedData.appendData(newData); - - - if (this.fields[i].filename) { - const fileData = NSData.alloc().initWithContentsOfFile(this.fields[i].filename); - combinedData.appendData(fileData); - results = CRLF; - } - - } - // Add final part of it... - results += "--" + this.boundary + "--" + CRLF; - tempString = NSString.stringWithString(results); - newData = tempString.dataUsingEncoding(NSUTF8StringEncoding); - combinedData.appendData(newData); - - NSFileManager.defaultManager.createFileAtPathContentsAttributes(fileName, combinedData, null); - - return fileName; - } - - public getHeader(): string { - return this.header; - } + private boundary: string; + private header: any; + private fileCount: number; + private fields: Array; + + constructor() { + this.clear(); + } + + public clear(): void { + this.boundary = '--------------formboundary' + Math.floor(Math.random() * 100000000000); + this.header = { 'Content-Type': 'multipart/form-data; boundary=' + this.boundary }; + this.fileCount = 0; + this.fields = []; + } + + public appendParam(name: string, value: string, filename?: string, mimeType?: string, destFileName?: string): void { + // If all we are doing is passing a field, we just add it to the fields list + if (filename == null) { + this.fields.push({ name: name, value: value }); + return; + } + // Load file + mimeType = mimeType || 'application/data'; + + if (filename.startsWith('~/')) { + filename = filename.replace('~/', knownFolders.currentApp().path + '/'); + } + + const finalName = destFileName || filename.substr(filename.lastIndexOf('/') + 1, filename.length); + this.fields.push({ name: name, filename: filename, destFilename: finalName, mimeType: mimeType }); + } + + public generateFile(): string { + const CRLF = '\r\n'; + + const fileName = knownFolders.documents().path + '/temp-MPF-' + Math.floor(Math.random() * 100000000000) + '.tmp'; + + const combinedData = NSMutableData.alloc().init(); + + let results: string = ''; + let tempString: NSString; + let newData: any; + for (let i = 0; i < this.fields.length; i++) { + results += '--' + this.boundary + CRLF; + results += 'Content-Disposition: form-data; name="' + this.fields[i].name + '"'; + if (!this.fields[i].filename) { + results += CRLF + CRLF + this.fields[i].value + CRLF; + } else { + results += '; filename="' + this.fields[i].destFilename + '"'; + if (this.fields[i].mimeType) { + results += CRLF + 'Content-Type: ' + this.fields[i].mimeType; + } + results += CRLF + CRLF; + } + + tempString = NSString.stringWithString(results); + results = ''; + newData = tempString.dataUsingEncoding(NSUTF8StringEncoding); + combinedData.appendData(newData); + + if (this.fields[i].filename) { + const fileData = NSData.alloc().initWithContentsOfFile(this.fields[i].filename); + combinedData.appendData(fileData); + results = CRLF; + } + } + // Add final part of it... + results += '--' + this.boundary + '--' + CRLF; + tempString = NSString.stringWithString(results); + newData = tempString.dataUsingEncoding(NSUTF8StringEncoding); + combinedData.appendData(newData); + + NSFileManager.defaultManager.createFileAtPathContentsAttributes(fileName, combinedData, null); + + return fileName; + } + + public getHeader(): string { + return this.header; + } }