[英]Meteor: Http Call return Undefined on response after an Http Response
我嘗試通過使用流星應用程序和自定義php腳本將內容編碼為base64來上傳文件。
php腳本如下:
require_once '../vendor/autoload.php';
use WindowsAzure\Common\ServicesBuilder;
use MicrosoftAzure\Storage\Common\ServiceException;
use MicrosoftAzure\Storage\Blob\Models\ListBlobsOptions;
error_log("Method:".$_SERVER['REQUEST_METHOD'],0);
if($_SERVER['REQUEST_METHOD'] === 'OPTIONS'){
header('Access-Control-Allow-Origin: *');
header('Access-Control-Allow-Methods: POST');
header('Access-Control-Allow-Headers: Origin, Content-Type, X-Auth-Token , Authorization');
error_log("Options Called",0);
die();
} else {
error_log("Post Called",0);
function create_storage_connection()
{
return "DefaultEndpointsProtocol=https;AccountName=".getenv('AZURE_ACCOUNT').";AccountKey=".getenv('AZURE_KEY');
}
$connectionString=create_storage_connection();
$blobRestProxy= ServicesBuilder::getInstance()->createBlobService($connectionString);
$container_name=getenv('AZURE_CONTAINER');
$data=file_get_contents('php://input');
$data=json_decode($data,true);
try{
//Upload data
$file_data=base64_decode($data['data']);
$data['name']=uniqid().$data['name'];
$blobRestProxy->createBlockBlob($container_name,$data['name'],$file_data);
$blob = $blobRestProxy->getBlob($container_name, $data['name']);
//Download url info
$listBlobsOptions = new ListBlobsOptions();
$listBlobsOptions->setPrefix($data['name']);
$blob_list = $blobRestProxy->listBlobs($container_name, $listBlobsOptions);
$blobs = $blob_list->getBlobs();
$url=[];
foreach($blobs as $blob)
{
$urls[]=$blob->getUrl();
}
error_log("Urls:\n".implode(" , ",$urls),0);
header("Content-type: application/json");
$result=json_encode(['files'=>"sent",'url'=>$urls]);
error_log("Result: ".$result,0);
echo $result;
} catch(ServiceException $e) {
$code = $e->getCode();
$error_message = $e->getMessage();
header("Content-type: application/json");
echo json_encode(['code'=>$code,'message'=>$error_message]);
}
}
在流星腳本上,我創建了一個名為“ imports / ui / File.jsx”的文件,其內容如下:
import React, { Component } from 'react';
import {FileUpload} from '../api/FileUpload.js';
class File extends Component {
changeFile(e) {
e.preventDefault()
let files = document.getElementById('fileUpload');
var file = files.files[0];
var reader=new FileReader();
reader.onloadend = function() {
Meteor.call('fileStorage.uploadFile',reader.result,file.name,file.type)
}
reader.readAsDataURL(file);
}
render() {
return (
<form onSubmit={ this.changeFile.bind(this) }>
<label>
<input id="fileUpload" type="file" name="file" />
</label>
<button type="submit">UploadFile</button>
</form>
)
}
}
export default File;
而且我還有一個名為imports/api/FileUpload.js
的文件,用於處理對服務器的http調用:
import { Meteor } from 'meteor/meteor';
import { HTTP } from 'meteor/http'
export default Meteor.methods({
'fileStorage.uploadFile'(base64Data,name,mime) {
// this.unblock();
let http_obj={
'data':{
'data':base64Data,
'name':name,
'mime':mime
},
}
HTTP.call("POST","http://localhost/base64Upload/",http_obj,function(err,response){
console.log("Response:",response);
});
}
});
問題是,即使我從服務器成功獲得以下響應:
console.log("Response:",response);
不將服務器腳本返回的json響應打印到控制台。 相反,我在瀏覽器控制台中收到以下消息:
回應:未定義
我無法理解為什么即使php腳本返回了響應,我在響應時仍未定義。 另外,如果我console.log
錯誤,我得到以下信息:
錯誤錯誤:網絡Καταγραφήστοίβας:httpcall_client.js / HTTP.call / xhr.onreadystatechange @ http:// localhost:3000 / packages / http.js?hash = d7408e6ea3934d8d6dd9f1b49eab82ac9f6d8340:244:20
我不知道為什么會這樣。
流星應用程序使用OPTIONS
方法執行2個Http調用,使用POST
根據要求,將die()
替換為:
var_dump($_SERVER['REQUEST_METHOD']); exit;
我得到答復:
/home/pcmagas/Kwdikas/php/apps/base64Upload/src/public/index.php:14:string'OPTIONS'
還在瀏覽器的“網絡”標簽上顯示:
請記住,流星對腳本執行2次http調用,一次使用http OPTIONS
方法進行,另一次使用http POST
。 我要獲得的是使用http POST
一種。
我還嘗試通過將http_obj
更改為http_obj
來設置2秒鍾的超時:
let http_obj={
'data':{
'data':base64Data,
'name':name,
'mime':mime
},
'timeout':2000
}
但是我收到以下錯誤:
錯誤錯誤:無法在模擬中設置計時器
最后,我需要使該方法在服務器上運行:
我通過將imports/api/FileUpload.js
更改為以下內容來做到這一點:(我還刪除了不需要的代碼)
import { Meteor } from 'meteor/meteor';
import { HTTP } from 'meteor/http'
export const UploadedFile=null;
if(Meteor.isServer){
Meteor.methods({
'fileStorage.uploadFile'(base64Data,name,mime) {
// this.unblock();
let http_obj={
'data':{
'data':base64Data,
'name':name,
'mime':mime
},
// 'timeout':2000,
'headers':{
'Content-Type': 'application/json'
}
}
return HTTP.call("POST","http://localhost/base64Upload/",http_obj);
}
});
}
並將此需求放入server/main.js
結果如下:
import { Meteor } from 'meteor/meteor';
import {FileUpload} from '../imports/api/FileUpload.js';
Meteor.startup(() => {
// code to run on server at startup
});
同樣在imports/ui/File.jsx
我這樣調用方法:
Meteor.call('fileStorage.uploadFile',reader.result,file.name,file.type,function(err,response){
console.log(response);
})
}
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.