【问题标题】:Python Twilio Error - Content is not allowed in prolog. Schema validation warningPython Twilio 错误 - prolog 中不允许有内容。架构验证警告
【发布时间】:2020-10-03 21:48:26
【问题描述】:

类似于之前的SO query,我有一个监听 POST 的 webhook 接收器。我向我的 Twilio 号码发送 WhatsApp 消息,Twilio POST 到 webhook 接收器,服务器代码处理请求,然后将响应对象返回给 Twilio。该代码从 Airtable 上的事件日历中提取一些项目,并应通过 Twilio 将它们发送到 WhatsApp。

Webhook 使用 Postman 测试正常。但是,WhatsApp 不返回 POST 请求的结果,并且我从 Twilio 调试器收到以下警告:

sourceComponent     "14100"
line                "1"
ErrorCode           "12200"
LogLevel            "WARN"
Msg                 "Content is not allowed in prolog."
EmailNotification   "false"
parserMessage       "Content is not allowed in prolog."
cols                "1"

我的完整代码在这里:

from flask import Flask, request
import requests
import json
from airtable import Airtable
from datetime import datetime
from twilio.twiml.messaging_response import MessagingResponse

base_key = 'BASE_KEY'
table_name = 'Events Calendar'
api_key = 'API_KEY'
airtable = Airtable(base_key, table_name, api_key)

API_URL = "https://api.airtable.com/v0/BASE_KEY/Events%20Calendar?maxRecords=3&filterByFormula=IS_AFTER(%7BDate%7D,NOW())"

headers = {
    "Authorization": "Bearer API_KEY"
}

app = Flask(__name__)

@app.route('/bot', methods=['POST'])
def accessdb():
    incoming_msg = request.values.get('Body', '').lower()
    resp = MessagingResponse()
    msg = resp.message()
    responded = False
    
    if 'next' in incoming_msg:
        def pretty(d):
            date = datetime.strptime(d["Date"], "%Y-%m-%dT%H:%M:%S.%fZ")
            return f'''Date: {date.strftime("%A, %d. %B %Y %I:%M%p")}
Title: {d['Title']}
Description: {d['Description']}
'''
        pages = airtable.get_iter(maxRecords=3, formula="Date >= NOW()", sort=["Date"], fields=('Date', 'Title', 'Description'))
        mylist = [] 
        
        for page in pages:
            for record in page:
                if 'fields' not in record:
                    continue
                fields = record['fields']
                mylist.append(pretty(fields))
        return "\n".join(mylist)

    elif 'what' in incoming_msg:
        msg.body("Please enter 'next' to get the next events.")
        responded = True

    elif not responded:
        msg.body("I don't know about that, sorry!")
    return str(resp)

if __name__ == '__main__':
    app.run()

以上代码部署到 Heroku,我可以使用 curl 或 Postman 调用 webhook 并成功产生以下结果:

Date: Wednesday, 07. October 2020 06:00PM
Title: Social Distancing Dance Party: Slow Motion
Description: Slow Motion is the yang to the Saturday Dance Party's yin

Date: Thursday, 08. October 2020 08:00AM
Title: Free Online Meditation Mornings
Description: 20-30 minute meditation. Donations can be made at https://example.com

Date: Saturday, 10. October 2020 07:00PM
Title: Social Distancing Dance Party
Description: A party in your very own kitchen

但是从 Twilio 调用 webhook 会产生之前给出的错误。

较早的SO答案建议返回查询结果如下:return twilioResponse.ToString()

我对语法没有信心,并尝试了以下方法:

        for page in pages:
            for record in page:
                if 'fields' not in record:
                    continue
                fields = record['fields']
                mylist.append(pretty(fields))
        return "\n".join.twilioResponse.ToString(mylist)

但是,当我尝试上述方法时,失败如下:

    return "\n".join.twilioResponse.ToString(mylist)
AttributeError: 'builtin_function_or_method' object has no attribute 'twilioResponse'

非常感谢您朝正确的方向轻推。非常感谢。

【问题讨论】:

  • 你能分享更多你的代码吗?目前尚不清楚您从哪里获取一些变量或从哪里返回 webhook 响应。
  • 谢谢@philnash - 我添加了更多信息。非常感谢。

标签: python python-3.x twilio twilio-api


【解决方案1】:

这里是 Twilio 开发者宣传员。

我认为这里的问题是,当您发送机器人“下一个”时,它会响应一串结果,而不是 TwiML,这是 Twilio 所期望的。如果用户发送“what”或其他任何内容,您将返回 TwiML,因此没有太多需要更改。

您需要获取此代码:

        for page in pages:
            for record in page:
                if 'fields' not in record:
                    continue
                fields = record['fields']
                mylist.append(pretty(fields))
        return "\n".join(mylist)

不要将return 放在末尾,而是将您构建的字符串设置为 TwiML 响应中的消息,如下所示:

        for page in pages:
            for record in page:
                if 'fields' not in record:
                    continue
                fields = record['fields']
                mylist.append(pretty(fields))
        msg.body("\n".join(mylist))

现在您将事件列表设置为要发送的消息,它将通过方法末尾的 return str(resp) 行作为 TwiML 返回到 Twilio。

顺便说一句,我认为您不需要在整个条件句中使用 responded 布尔值。相反,您可以使用else 完成您的条件,这将捕获您之前未匹配的任何情况。像这样:

    else:
        msg.body("I don't know about that, sorry!")

让我知道这是否有帮助。

【讨论】:

  • 谢谢@philnash - 是的,这行得通。惊人的。我将查看有关 responded 布尔值的代码 - 我已改编自 here 的 Flask/Twilio 教程。这是一次这样的旅程和学习经历。再次感谢您。
  • 一个(希望如此)快速问题@philnash,虽然我引起了你的注意 - WhatsApp '有帮助' 扩展消息正文中的任何 URL。有没有办法抑制这种行为,因为它会干扰消息的格式?
  • 您可以用反引号将 URL 括起来以使其呈现为代码,这可能会抑制 URL 展开。虽然我没有尝试过,所以我不能保证它有效。见this article for details on text formatting in WhatsApp
  • 非常感谢@philnash - 我会看看那个。一切顺利。
猜你喜欢
  • 1970-01-01
  • 2019-03-09
  • 2023-04-06
  • 1970-01-01
  • 1970-01-01
  • 2015-12-15
  • 1970-01-01
  • 2019-04-04
  • 2019-04-23
相关资源
最近更新 更多