Dart json.encode不是Firebase函数所需的编码

时间:2018-05-04 19:33:47

标签: json dart google-cloud-functions

我已经在这个问题上工作了一段时间,而我似乎无法弄清楚究竟是什么问题。在Dart(2)中,json.encode()似乎没有给我我想要的结果。

我传递的Map<String, dynamic>看起来像这样:

_data = <String, dynamic>{
    'uid': newProfileCreationModel.accountID,
    'displayName': newProfileCreationModel.profileName,
    'greeting': newProfileCreationModel.profileBody ?? '',
    'avatarSrc': newProfileCreationModel.avatarSrc,
    'heroSrc': newProfileCreationModel.heroSrc,
    'accountType': newProfileCreationModel.accountType,
    'cityID': newProfileCreationModel.cityID,
    'cityName': newProfileCreationModel.cityName,
    'country': newProfileCreationModel.countryCode,
    'state': newProfileCreationModel.stateAbv,
  };

并使用它将其转换为JSON

final String _jsonData = json.encode(_data);

然后我将通过以下方式将其发送到Google云端功能。

final Uri _uri = Uri.parse(functionUrl);
final String _jsonData = json.encode(_data);
final String _userToken = await AuthenticationService.getUserToken();

HttpClient client = new HttpClient();
HttpClientRequest request = await client.postUrl(_uri);
request.headers.set('Authorization', 'Bearer ' + _userToken);
request.headers.set('Content-Type', 'application/json; charset=utf-8');
print('Encoded: ' + _jsonData);
request.write(_jsonData);

HttpClientResponse response = await request.close();
if(response.statusCode != HttpStatus.OK){ ...

我打印编码字符串的行将其输出到控制台:

05-04 18:52:57.902 26146-26200/com.app.name I/flutter: Encoded: {"uid":'123456789',"displayName":"James","greeting":"My Greetings!","avatarSrc":"http://cdn.free.com/someImage.jpg","heroSrc":"http://cdn.free.com/someImage.jpg","accountType":"per","cityID":1,"cityName":"Eugene","country":"US","state":"OR"}

但是request.write(_jsonData)失败并出现以下firebase控制台日志错误请求正文缺失数据

中的响应如下所示

Firebase控制台日志。

Request body is missing data.  { 
    uid: '123456789',
    displayName: 'James',
    greeting: 'My Greetings!',
    avatarSrc: 'http://cdn.free.com/someImage.jpg',
    heroSrc: 'http://cdn.free.com/someImage.jpg',   accountType: 'per',   
    cityID: 1,
    cityName: 'Eugene',   
    country: 'US',
    state: 'OR' 
}


Invalid request IncomingMessage {
  _readableState: 
   ReadableState {
     objectMode: false,
     highWaterMark: 16384,
     buffer: BufferList { head: null, tail: null, length: 0 },
     length: 0,
     pipes: null,
     pipesCount: 0,
     flowing: true,
     ended: true,
     endEmitted: true,
     reading: false,
     sync: false,
     needReadable: false,
     emittedReadable: false,
     readableListening: false,
     resumeScheduled: false,
     defaultEncoding: 'utf8',
     ranOut: false,
     awaitDrain: 0,
     readingMore: false,
     decoder: null,
     encoding: null },
  readable: false,
  domain: null,
  _events: {},
  _eventsCount: 0,
  _maxListeners: undefined,
  socket: 
   Socket {
     connecting: false,
     _hadError: false,
     _handle: 
      TCP {
        bytesRead: 13285,
        _externalStream: {},
        fd: 14,
        reading: true,
        owner: [Circular],
        onread: [Function: onread],
        onconnection: null,
        writeQueueSize: 0,
        _consumed: true },
     _parent: null,
     _host: null,

有趣的是,数据正在通过firebase控制台日志中清晰显示,但它不会将其识别为正文。

原始数据方法

当我尝试通过request.write()作为

发送原始JSON对象时
request.write({'hello':'universe'});

我在firebase控制台中遇到了完全不同的错误。

SyntaxError: Unexpected token h in JSON at position 1
    at Object.parse (native)
    at parse (/var/tmp/worker/node_modules/body-parser/lib/types/json.js:84:17)
    at /var/tmp/worker/node_modules/body-parser/lib/read.js:102:18
    at IncomingMessage.onEnd (/var/tmp/worker/node_modules/raw-body/index.js:149:7)
    at IncomingMessage.g (events.js:292:16)
    at emitNone (events.js:86:13)
    at IncomingMessage.emit (events.js:185:7)
    at endReadableNT (_stream_readable.js:974:12)
    at _combinedTickCallback (internal/process/next_tick.js:80:11)
    at process._tickDomainCallback (internal/process/next_tick.js:128:9)
在firebase端的

我正在使用一个使用callable的函数,这是从

记录firebase控制台日志的地方
export const finalizeProfile = functions.https.onCall((data, context) => { 
    //CODE 
});

是否有人能够发现我可能做错了什么?

2 个答案:

答案 0 :(得分:4)

解决方案隐藏在视线之内。这个问题与缺失的领域有关。对于Firebase云功能,错误消息data中提到的内容&#39;字面意思是它需要一个名为functions.https.onCall()的密钥,其中包含您传递的数据对象

谷歌文档在这方面并不清楚(因为它们缺少一个例子https://firebase.google.com/docs/functions/callable-reference#request_body

这是必须将数据发送到{ "data":{ "uid":"123456789", "displayName":"James", "greeting":"My Greeting!", "avatarSrc":"http://cdn.free.com/someImage.jpg", "heroSrc":"http://cdn.free.com/someImage.jpg", "accountType":"per", "cityID":1, "cityName":"Eugene", "country":"OR", "state":"AL" } } 的方式,请注意数据字段与原始问题的比较,而不包括此内容:

// Helper function to format any map to be used for Firebase
Map prepMapForFirebaseJSON(Map<String, dynamic> mapData){
    Map<String, Map<String, dynamic>> _fireJson = {
      'data' : mapData
    };
    return _fireJson;
}

// Process HTTP request
final Uri _uri = Uri.parse(functionUrl);
final String _jsonData = json.encode(prepMapForFirebaseJSON(mapData));
final String _userToken = await AuthenticationService.getUserToken();

HttpClient client = new HttpClient();
HttpClientRequest request = await client.postUrl(_uri);
request.headers.set('Authorization', 'Bearer ' + _userToken);
request.headers.set('Content-Type', 'application/json; charset=utf-8');
request.write(_jsonData);
request.close();

现在可以使用的结果代码如下所示:

{{1}}

答案 1 :(得分:0)

我认为没有关闭请求:

request.write(_jsonData);
var response = await request.close();