javascript - 对 Firebase Cloud Functions(Spark 层)中外部 API 的 HTTP 请求被拒绝
问题描述
我正在尝试从 Web 调用外部资源并使用 NodeJS Client V2 和 Cloud Functions 将结果加载到 Dialogflow 中。
我已经尝试了此代码的多种组合,使用承诺、外部函数等。没有运气。
测试1
function myIntent(conv) {
// console.log('conv ', conv)
const config ={
"headers": {'X-App-Token': dataSFAPITOken},
"Content-Type": "application/json"
}
const url = "http://data.sfgov.org/resource/cuks-n6tp.json";
return new Promise((resolve, reject) => {
axios
.get(url)
.then(response => {
console.log(response);
conv.ask('I got your data back')
return resolve(response)
})
.catch(error => {
console.log(error);
conv.ask('Sorry! Something went wrong')
return reject(error)
});
})
}
app.intent('My Intent', myIntent);
exports.myAgent = functions.https.onRequest(app);
错误
这是我在调用myIntent
.
{ Error: getaddrinfo ENOTFOUND data.sfgov.org data.sfgov.org:80
at errnoException (dns.js:28:10)
at GetAddrInfoReqWrap.onlookup [as oncomplete] (dns.js:76:26)
code: 'ENOTFOUND',
errno: 'ENOTFOUND',
syscall: 'getaddrinfo',
hostname: 'data.sfgov.org',
host: 'data.sfgov.org',
port: 80,
config:
{ adapter: [Function: httpAdapter],
transformRequest: { '0': [Function: transformRequest] },
transformResponse: { '0': [Function: transformResponse] },
timeout: 0,
xsrfCookieName: 'XSRF-TOKEN',
xsrfHeaderName: 'X-XSRF-TOKEN',
maxContentLength: -1,
validateStatus: [Function: validateStatus],
headers:
{ Accept: 'application/json, text/plain, */*',
'User-Agent': 'axios/0.18.0' },
method: 'get',
url: 'http://data.sfgov.org/resource/cuks-n6tp.json',
data: undefined },
request:
Writable {
_writableState:
WritableState {
objectMode: false,
highWaterMark: 16384,
needDrain: false,
ending: false,
ended: false,
finished: false,
decodeStrings: true,
defaultEncoding: 'utf8',
length: 0,
writing: false,
corked: 0,
sync: true,
bufferProcessing: false,
onwrite: [Function],
writecb: null,
writelen: 0,
bufferedRequest: null,
lastBufferedRequest: null,
pendingcb: 0,
prefinished: false,
errorEmitted: false,
bufferedRequestCount: 0,
corkedRequestsFree: [Object] },
writable: true,
domain: null,
_events:
{ response: [Function: handleResponse],
error: [Function: handleRequestError] },
_eventsCount: 2,
_maxListeners: undefined,
_options:
{ protocol: 'http:',
maxRedirects: 21,
maxBodyLength: 10485760,
path: '/resource/cuks-n6tp.json',
method: 'get',
headers: [Object],
agent: undefined,
auth: undefined,
hostname: 'data.sfgov.org',
port: null,
nativeProtocols: [Object],
pathname: '/resource/cuks-n6tp.json' },
_redirectCount: 0,
_requestBodyLength: 0,
_requestBodyBuffers: [],
_onNativeResponse: [Function],
_currentRequest:
ClientRequest {
domain: null,
_events: [Object],
_eventsCount: 6,
_maxListeners: undefined,
output: [],
outputEncodings: [],
outputCallbacks: [],
outputSize: 0,
writable: true,
_last: true,
upgrading: false,
chunkedEncoding: false,
shouldKeepAlive: false,
useChunkedEncodingByDefault: false,
sendDate: false,
_removedHeader: {},
_contentLength: 0,
_hasBody: true,
_trailer: '',
finished: true,
_headerSent: true,
socket: [Object],
connection: [Object],
_header: 'GET /resource/cuks-n6tp.json HTTP/1.1\r\nAccept: application/json, text/plain, */*\r\nUser-Agent: axios/0.18.0\r\nHost: data.sfgov.org\r\nConnection: close\r\n\r\n',
_headers: [Object],
_headerNames: [Object],
_onPendingData: null,
agent: [Object],
socketPath: undefined,
timeout: undefined,
method: 'GET',
path: '/resource/cuks-n6tp.json',
_ended: false,
_redirectable: [Circular],
parser: null },
_currentUrl: 'http://data.sfgov.org/resource/cuks-n6tp.json' },
response: undefined }
测试2
const url = "data.sfgov.org";
var options = {
protocol:'http:',
host: url,
port : 8080,
path:'/resource/cuks-n6tp',
headers: {
'X-App-Token': dataSFAPITOken,
"Content-Type": "application/json"
}
};
http.get(options, (http_res) => {
// initialize the container for our data
var data = "";
// this event fires many times, each time collecting another piece of the response
http_res.on("data", (chunk) =>{
// append this chunk to our growing `data` var
data += chunk;
});
// this event fires *one* time, after all the `data` events/chunks have been gathered
http_res.on("end", () => {
// you can use res.send instead of console.log to output via express
console.log(data);
});
});
这是官方文档的链接HERE。
疯狂的是,下面是浏览器上的类似实现,它可以工作。事实上,我不需要 API 密钥来查询数据。我可以简单地复制/粘贴链接。
$.ajax({
url: "https://data.sfgov.org/resource/cuks-n6tp.json",
type: "GET",
data: {
"$limit" : 50,
"$$app_token" : "APPTOKEN"
}
}).done(function(data) {
// alert("Retrieved " + data.length + " records from the dataset!");
console.log(data);
document.getElementById('data').innerHTML = JSON.stringify(data, null,2)
});
但由于某些原因,我无法使其与 Dialogflow NodeJS Client V2 一起使用。我很确定在 V1 中,我可以让它工作。
我迁移到 V2 有点痛苦。请帮忙。
谢谢。
解决方案
Firebase Cloud Functions 上的免费Spark
计划不允许调用 Google 以外的域。
您将需要升级到其中一个付费计划,例如Flame
orBlaze
计划,这需要信用卡存档。但是,对于低水平的使用,您不会被收取费用Blaze
。
下图显示Spark
、Flame
和Blaze
。注意Google services only
for Spark
。https://firebase.google.com/pricing/
2020 年 11 月更新
请注意,该Flame
计划不再可用。因此,如果您想要外部访问,则需要使用该Blaze
计划。
推荐阅读
- mysql - 内连接的执行速度是否与 where 子句一样快?
- c# - `Model.plans.Count` 引发了“System.NullReferenceException”类型的异常
- mysql - 如何在mysql的select语句中复用多列的值
- maximo - 在 Maximo 查找字段中显示相关值?
- c - 虽然当我尝试比较 2 个 int 值(1 个来自用户输入,1 个来自数据 .txt 文件)时不起作用(学生记录项目)
- linux - Docker:从 CLI 运行容器失败,但不是从 docker 桌面运行,不允许 RTENETLINK 操作
- javascript - 如何在不打开新浏览器窗口的情况下使用链接打开外部应用程序?
- ruby-on-rails - 如何设置 Sidekiq 来处理一个类的 Webhook
- python - 相互比较字典列表的所有 url
- c# - 角度应用程序正在获得访问权限,但从另一个系统调用时未访问 webapi,但在服务器内尝试工作。为什么?