为什么我的ZenDesk宏会被更新,但实际上没有任何变化?

时间:2016-08-12 17:01:17

标签: python python-2.7 zendesk-api

我试图在ZenDesk上批量编辑我的个人宏的签名,唯一的方法是通过API。所以我写了这个快速的Python脚本来尝试这样做:

import sys
import time
import logging
import requests
import re

start_time = time.time()

# Set up logging
logger = logging.getLogger()
log_handler = logging.StreamHandler(sys.stdout)
log_handler.setFormatter(logging.Formatter("%(asctime)s - %(name)s - %(levelname)s - %(message)s - %(funcName)s - line %(lineno)d"))
log_handler.setLevel(logging.DEBUG)
logger.addHandler(log_handler)
logger.setLevel(logging.DEBUG)

def doTheGet(url, user, pwd):
        response = requests.get(url, auth=(user + "/token", pwd))

        if response.status_code != 200:
                logger.error("Status: %s (%s) Problem with the request. Exiting. %f seconds elapsed" % (response.status_code, response.reason, time.time() - start_time))
                exit()

        data = response.json()
        return data

def doThePut(url, updated_data, user, pwd):
        response = requests.put(url, json="{'macro': {'actions': %r}}" % updated_data, headers={"Content-Type": "application/json"}, auth=(user + "/token", pwd))

        if response.status_code != 200:
                logger.error("Status: %s (%s) Problem with the request. Exiting. %f seconds elapsed" % (response.status_code, response.reason, time.time() - start_time))
                exit()

        data = response.json()
        return data

def getMacros():
        macros = {}

        data = doTheGet("https://mydomain.zendesk.com/api/v2/macros.json", "me@mydomain.com", "111tokenZZZ")

        def getMacros(macro_list, page, page_count):
                if not page:
                        for macro in macro_list:
                                if macro["restriction"] and macro["active"]:
                                        if macro["restriction"]["type"] == "User":
                                                macros[macro["id"]] = macro["actions"]
                else:
                        for macro in macro_list:
                                if macro["restriction"] and macro["active"]:
                                        if macro["restriction"]["type"] == "User":
                                                macros[macro["id"]] = macro["actions"]

                        page_count += 1
                        new_data = doTheGet(page, "me@mydomain.com", "111tokenZZZ")
                        new_macs = new_data["macros"]
                        new_next_page = new_data["next_page"]
                        getMacros(new_macs, new_next_page, page_count)


        macs = data["macros"]
        current_page = 1
        next_page = data["next_page"]
        getMacros(macs, next_page, current_page)
        return macros

def updateMacros():
        macros = getMacros()

        regular = "RegEx to match signature to be replaced$" #since some macros already have the updated signature

        for macro in macros:
                for action in macros[macro]:
                        if action["field"] == "comment_value":
                                if re.search(regular, action["value"][1]):
                                        ind = action["value"][1].rfind("\n")
                                        action["value"][1] = action["value"][1][:ind] + "\nNew signature"

        return macros

macs = updateMacros()

for mac in macs:
        doThePut("https://mydomain.zendesk.com/api/v2/macros/%d.json" % (mac), macs[mac], "me@mydomain.com", "111tokenZZZ")

现在,一切都按预期运行,我没有错误。当我在ZenDesk上访问我的宏并通过最后更新对它们进行排序时,我确实看到脚本做了一些事情,因为它们显示为今天的最后更新。但是,它们没有任何变化。我确保通过发送的数据已经编辑(updateMacros 正在执行其工作)。我确保请求发回一个OK响应。所以我发送更新的数据,得到200响应,但是the response sent back向我展示了之前的宏,没有任何变化。

我发现的唯一可能在某种程度上错误的是我发送的数据的格式,或类似的东西。但即便如此,我还是期待回应不是200,那么......

我在这里缺少什么?

1 个答案:

答案 0 :(得分:5)

看起来你在PUT请求中对JSON数据进行了双重编码:

response = requests.put(url, json="{'macro': {'actions': %r}}" % updated_data, headers={"Content-Type": "application/json"}, auth=(user + "/token", pwd))

json参数需要一个对象,然后它尽职地编码为JSON并作为请求的主体发送;这只是一种便利;实施很简单,

    if not data and json is not None:
        # urllib3 requires a bytes-like body. Python 2's json.dumps
        # provides this natively, but Python 3 gives a Unicode string.
        content_type = 'application/json'
        body = complexjson.dumps(json)
        if not isinstance(body, bytes):
            body = body.encode('utf-8')

(来源:https://github.com/kennethreitz/requests/blob/master/requests/models.py#L424

由于值始终通过json.dumps()传递,如果传递表示已编码JSON的字符串,则它本身将被编码:

"{\'macro\': {\'actions\': [{\'field\': \'comment_value\', \'value\': [\'channel:all\', \'Spiffy New Sig that will Never Be Saved\']}]}}"

ZenDesk在获得JSON时没有预料到,会更新updated_at字段并且......别无其他。您可以通过传递一个空字符串来验证这一点 - 结果相同。

请注意,您还依赖Python的repr格式来填充您的JSON;这也许是一个坏主意。相反,让我们重新构建我们的宏对象并让请求对其进行编码:

response = requests.put(url, json={'macro': {'actions': updated_data}}, headers={"Content-Type": "application/json"}, auth=(user + "/token", pwd))

这应该符合您的期望。