Отправка JSON с помощью тестового клиента django - PullRequest
35 голосов
/ 21 декабря 2011

Я работаю над django проектом, который будет служить конечной точкой для webhook.Webhook отправит некоторые данные JSON в мою конечную точку, которая затем проанализирует эти данные.Я пытаюсь написать модульные тесты для него, но я не уверен, правильно ли я отправляю JSON.

Я продолжаю получать «TypeError: строковые индексы должны быть целыми числами» в pipe_endpoint

Вот код:

# tests.py
from django.test import TestCase
from django.test.client import Client
import simplejson

class TestPipeline(TestCase):

    def setUp(self):
        """initialize the Django test client"""
        self.c = Client()

    def test_200(self):
        json_string = u'{"1": {"guid": "8a40135230f21bdb0130f21c255c0007", "portalId": 999, "email": "fake@email"}}'
        json_data = simplejson.loads(json_string)
        self.response = self.c.post('/pipeline-endpoint', json_data, content_type="application/json")
        self.assertEqual(self.response.status_code, "200")

и

# views.py
from pipeline.prospect import Prospect
import simplejson

def pipeline_endpoint(request):

    #get the data from the json object that came in
    prospects_json = simplejson.loads(request.raw_post_data)
    for p in prospects_json:
        prospect = {
            'email'          : p['email'],
            'hs_id'          : p['guid'],
            'portal'         : p['portalId'],
        }

Редактировать: весь traceback.

======================================================================
ERROR: test_200 (pipeline.tests.TestPipeline)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "F:\......\pipeline\tests.py", line 31, in test_200
    self.response = self.c.post('/pipeline-endpoint', json_string, content_type="application/json")
  File "C:\Python27\lib\site-packages\django\test\client.py", line 455, in post
    response = super(Client, self).post(path, data=data, content_type=content_type, **extra)
  File "C:\Python27\lib\site-packages\django\test\client.py", line 256, in post
    return self.request(**r)
  File "C:\Python27\lib\site-packages\django\core\handlers\base.py", line 111, in get_response
    response = callback(request, *callback_args, **callback_kwargs)
  File "F:\......\pipeline\views.py", line 18, in pipeline_endpoint
    'email'          : p['email'],
TypeError: string indices must be integers

----------------------------------------------------------------------
Ran 1 test in 0.095s

FAILED (errors=1)
Destroying test database for alias 'default'...

Ответы [ 6 ]

57 голосов
/ 12 августа 2013

@ mrmagooey прав

def test_your_test(self):
    python_dict = {
        "1": {
            "guid": "8a40135230f21bdb0130f21c255c0007",
            "portalId": 999,
            "email": "fake@email"
        }
    }
    response = self.client.post('/pipeline-endpoint/',
                                json.dumps(python_dict),
                                content_type="application/json")

используйте json.dumps вместо json.loads

13 голосов
/ 26 августа 2016

Попробуйте:

 self.client.generic('POST', '/url', json.dumps({'json': 'object'})
9 голосов
/ 06 апреля 2014

Вы всегда можете использовать HttpRequest.body , который загружает необработанные данные запроса.Таким образом, вы можете самостоятельно обрабатывать данные.

c = Client()
json_str= json.dumps({"data": {"id": 1}})
c.post('/ajax/handler/', data= json_str, content_type='application/json',
                                         HTTP_X_REQUESTED_WITH='XMLHttpRequest')


def index(request):
    ....
    print json.loads(request.body)
4 голосов
/ 08 декабря 2017

rest_framework 's APIClient (который является значением по умолчанию client_class в APITestCase) обеспечивает сброс dict до JSON и егоустанавливает правильный тип содержимого, передавая format='json'.

from rest_framework import status
from rest_framework.test import APIClient, APITestCase


class MyTestCase(APITestCase):
    url = '/url'

    def post(self, payload, url=None):
        """
        Helper to send an HTTP post.

        @param (dict) payload: request body

        @returns: response
        """
        if url is None:
            url = self.url

        return self.client.post(url, payload, format='json')

    def test_my_function(self):
        payload = {
            'key': 'value'
        }
        response = self.post(payload)
        self.assertEqual(response.status_code, status.HTTP_200_OK)
2 голосов
/ 21 декабря 2011

Вы можете использовать iteritems в словарях для цикла

for index, p in prospects_json.iteritems():
  prospect={
    'email': p['email'],
  }

или альтернативно

for index in prospect_json:
  prospect={
    'email': prospect_json[ index ]['email']
  }
0 голосов
/ 13 сентября 2018

В дополнение к ответу Гийома Винсента из Django 2.1 нам больше не нужно использовать json.dumps для передачи данных.

Изменено в Django 2.1: Описанная выше сериализация JSON была добавлена. В более старых версиях вы можете вызывать json.dumps () для данных, прежде чем передавать их в post (), чтобы добиться того же.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...