6

我想要完成的是使用 URLLoader 类和 URLRequest 将一些二进制数据,特别是表示 PNG 图像的 ByteArray 上传到服务器。

当我将contentTypeURLRequest 的属性设置为“multipart/form-data”而不是默认值时,调用会urlLoader.load()导致安全异常。

当我将该contentType属性保留为默认值时,它可以正常工作,但需要很长时间(与 PNG 文件的长度成正比)才能将文件上传到服务器。

所以,我的问题是为什么我会得到这个安全异常?我该如何避免呢?

请注意,我的 SWF 是从开发服务器而不是本地文件系统(准确地说是 Google App Engine 开发服务器)提供的。

这是代码:

var pngFile:ByteArray = PNGEncoder.encode(bitmapData);

var urlRequest:URLRequest = new URLRequest('/API/uploadImage');

// With this line of code, the call to urlLoader.load() throws the following security exception:
// 'SecurityError: Error #2176: Certain actions, such as those that display a pop-up window, may only be invoked upon user interaction, for example by a mouse click or button press.'
urlRequest.contentType = 'multipart/form-data';

urlRequest.method = URLRequestMethod.POST;
urlRequest.data = pngFile;
urlRequest.requestHeaders.push(new URLRequestHeader('Cache-Control', 'no-cache'));

urlLoader = new URLLoader();
urlLoader.dataFormat = URLLoaderDataFormat.TEXT;
urlLoader.addEventListener(Event.COMPLETE, onUploadComplete);
urlLoader.addEventListener(IOErrorEvent.IO_ERROR, onUploadError);

NextFrame.addCallback(function () {
    urlLoader.load(urlRequest);
});
4

3 回答 3

13

可能contentType不是指您发送的数据,而是指您接收的数据。尝试设置requestHeaders,这应该可以工作:

urlRequest.requestHeaders.push(new URLRequestHeader('Content-type', 'multipart/form-data'));

另外,我在我的一个项目中找到了一段代码。该代码工作并使用 POST 将一些二进制 JPEG 数据发送到服务器。我前段时间做过,我无法解释为什么我会这样做,但也许它会有所帮助。我按原样粘贴它:

function sendData(submitPath:String, descriere:String):void {
    // building the url request for uploading the jpeg to the server
    var header:URLRequestHeader = new URLRequestHeader('Content-type', 'application/octet-stream');
    var jpgURLRequest:URLRequest = new URLRequest(submitPath+'/id/'+player.id+'/path/'+player.contentPath.replace('/','')+'/width/'+player.videoWidth+'/height/'+player.videoHeight+'/descriere/'+descriere+'/timp/'+time);
    jpgURLRequest.requestHeaders.push(header);
    jpgURLRequest.method = URLRequestMethod.POST;
    jpgURLRequest.data = screenShot;

    // sending the data to the server
    var sender:URLLoader = new URLLoader();
    sender.load(jpgURLRequest);
}
于 2009-08-13T12:40:19.227 回答
8

为了完整起见,这就是我最终设置 URLRequest 对象的方式(其他一切都保持不变):

urlRequest.method = URLRequestMethod.POST;
urlRequest.data = UploadPostHelper.getPostData('filename', pngFile);
urlRequest.requestHeaders.push(new URLRequestHeader('Cache-Control', 'no-cache'));
urlRequest.requestHeaders.push(new URLRequestHeader('Content-Type', 'multipart/form-data; boundary=' + UploadPostHelper.getBoundary()));

正如 evilpenguin 所指出的,关键是根本不设置 contentType 属性,而是将其放在标题中。但是,仅使用“multipart/form-data”时,我在服务器端收到有关无效 POST 边界的错误,因此我最终使用了一个名为 UploadPostHelper 的类来为文件上传创建有效边界和 POST 正文。

这修复了神秘的安全错误(我仍然不知道为什么会这样),并且等待上传的时间很长。

应该注意的是,使用 UploadPostHelper 的示例代码涉及设置contentTypeURLRequest 对象的属性,这显然适用于某些人,但不适用于我的情况。

于 2009-08-13T18:36:46.527 回答
4

我遇到过同样的问题。提交到 PHP 脚本时它工作正常,但不是 ASP 脚本。将内容类型移动到 requestHeader 后,它可以正常工作。这是我的代码:

// Object containing form fields
var formdata = new Object();
formdata.Email = textArray[8].text;

//URLRequest containing the form fields and the attached image
var urlRequest : URLRequest = new URLRequest(url);
urlRequest.method = URLRequestMethod.POST;
urlRequest.data = UploadPostHelper.getPostData( imageName, imageByteArray, formdata );
urlRequest.requestHeaders.push( new URLRequestHeader( 'Cache-Control', 'no-cache' ) );
urlRequest.requestHeaders.push(new URLRequestHeader('Content-Type', 'multipart/form-data; boundary=' + UploadPostHelper.getBoundary())); 

//URLLoader to load the request
var urlLoader : URLLoader = new URLLoader();
urlLoader.dataFormat = URLLoaderDataFormat.BINARY;
urlLoader.load( urlRequest );
于 2009-11-18T16:36:59.173 回答