Firebase云功能返回304错误,然后重新启动

时间:2019-03-08 18:11:43

标签: javascript node.js firebase firebase-realtime-database google-cloud-functions

我已经坚持了几个月。我从函数中删除了一些次要细节,但没有什么大不了的。我有这个https云功能,可以结束会话,然后使用endTimestartTime来计算bill,然后将其返回给客户端。

startTime是从实时Firebase数据库(会话启动程序函数放在那里)中检索的。

我的代码段:

exports.endSession = functions.https.onRequest(async (req, res) => {
    console.log("endSession() called.")
    if(req.method == 'GET'){
        bid = req.query.bid
        session_cost = req.query.sessioncost
    }else{
        bid = req.body.bid
        session_cost = req.body.sessioncost
    }

start_time_ref = admin.database().ref("/online_sessions/").child(bid).child("start_time")
start_time_snapshot = await start_time_ref.once('value')

console.log("start_time_snapshot: "+start_time_snapshot.val())

start_time_snapshot = moment(start_time_snapshot.val(), 'dddd MMMM Do YYYY HH:mm:ss Z');
endDateTime = getDateTime()

console.log("startTime: " + start_time_snapshot.toString())
console.log("endTime: " + endDateTime.toString())
hour_difference = getHourDifference(start_time_snapshot, endDateTime)

bill = ride_cost * Math.ceil(hour_difference)
console.log("bill: "+bill)

var s_phone
sSessionlinks_ref = admin.database().ref('/sSessionlinks/')
sSessionlinks_snapshot = await sSessionlinks_ref.once('value')

sSessionlinks_snapshot.forEach((sid)=>{
    if(sid.val() == bid){
        s_phone = sid.key
    }
})

s_fcm_token_ref = admin.database().ref("/s/").child(s_phone).child("FCM")
s_fcm_token_snapshot = await s_fcm_token_ref.once('value')

try{ // telling another client that session has ended.
    await admin.messaging().send({
        data: {
            type: "sessionCompleted",
            bill: bill.toString()
        },
        token: s_fcm_token_snapshot.val() 
    })
}catch(error){

}

//deleting this session from online sessions
online_session_ref = admin.database().ref('/online_sessions/').child(bid)
await online_session_ref.remove()

//puting this session as available
available_session_ref = admin.database().ref('/available_sessions/')
json = {}
json[bid] = s_phone
await available_session_ref.update(json) // session made available


res.status(200).send(bill.toString())
// here it *sometimes* returns 304 and then restarts but since i've already removed online_session_ref I cannot get startTime again because its removed with online_sessions so it fails.
    // return
})

首次调用时。它会正确执行所有计算,但会返回304。因此(我认为客户端)会重新发送请求,并再次调用该函数,但由于会话被破坏,因此无法计算startTime

为什么在第一次调用时,即使所有计算正确进行,它都返回304而不是200?这个问题不会一直发生。通常会在很长一段时间后调用此函数,但是我不确定。我不知道是什么原因造成的。

我使用过的辅助函数:

function getHourDifference(s, e){
    return moment.duration(e.diff(s)).asHours()
}


function getDateTime(){
    d = moment.utc().utcOffset('+0530')
    return d
}

函数第一次结束时,文本有效载荷为Function execution took 794 ms, finished with status code 304

第二次运行时(无法获得startTime导致其在第一次运行中被删除。首先不应进行第二次运行。),有效载荷文本为{{1} }(其200,但返回Function execution took 234 ms, finished with status code 200,因为没有NaN便无法进行计算。

编辑: 正如你们中的一些人问我如何调用该函数:

使用Volley从android应用程序调用它。确保参数不为空。调用该函数的代码段为:

startTime

更新:@tuledev通过解决方法帮助修复了304,但是问题仍然存在。即使状态码为200,仍然以某种方式再次调用了云功能,但我得到了一张NaN帐单。此时,我不知道是什么原因造成的。

1 个答案:

答案 0 :(得分:2)

之所以出现304状态是因为响应与之前的响应相同。 Firebase云响应304和客户端将获取缓存的数据。

为防止出现304状态,我们可以返回账单的值+ uid,或者使响应与上一个有所不同。

正如我和OP所讨论的,304已解决,但问题仍然存在。所以我认为问题出在客户端。

希望有人可以帮助他。

编辑:OP在这里,我将客户端代码更改为使用okhttp而不是volley,经过2天的测试,一切似乎都很好。 Tuledev的答案确实解决了304,但即使在200之后,问题仍然存在。只需使用okhttp而不是volley。