Web2Py POST Failure - PullRequest
       29

Web2Py POST Failure

0 голосов
/ 22 сентября 2019

Сбой Web2Py POST

Размещение некоторых словарных данных в Web2py.Определенные данные POSTS последовательно успешно выполняются, а некоторые данные POSTS последовательно терпят неудачу.

Эти данные последовательно преуспевают в POST:

data = {"batt chrg": {
            "dat": "-999%",
            "measurement": "percent",
            "unit": "electric",
            "entrytype": "subunitStatus"
            },
         "pv gen": {
             "dat": "-999 W",
             "measurement": "level",
             "unit": "electric",
             "entrytype": "subunitStatus"
             }
         }

Эти данные (более длинный словарь) последовательно не в состоянии POST:

data = {"download": {
            "entrytype": "subunitStatus",
            "dat": "9 Mbit/s",
            "measurement": "level",
            "unit": "comms"},
        "batt chrg": {
            "entrytype": "subunitStatus",
            "dat": "94.4%",
            "measurement": "percent",
            "unit": "electric"},
         "pv gen": {
             "entrytype": "subunitStatus",
             "dat": "543 W",
             "measurement": "level",
             "unit": "electric"},
         "living space": {
             "entrytype": "subunitStatus",
             "dat": "72 F",
             "measurement": "level",
             "unit": "hvac"},
         "ping": {
             "entrytype": "subunitStatus",
             "dat": "60 ms",
             "measurement": "level",
             "unit": "comms"},
         "invert load": {
             "entrytype": "subunitStatus",
             "dat": "402 W",
             "measurement": "level",
             "unit": "electric"},
         "humidity": {
             "entrytype": "subunitStatus",
             "dat": "45%",
             "measurement": "level",
             "unit": "hvac"},
         "upload": {
             "entrytype": "subunitStatus",
             "dat": "8 Mbit/s",
             "measurement": "level",
             "unit": "comms"
             }
         }

Код Python на стороне клиента:

import requests
from requests.auth import HTTPBasicAuth
from datetime import datetime
import json

auth=HTTPBasicAuth('LoginID', 'Password')
unitid = '19700111'
uploaded = 'no'
data = {
    'unitid': unitid,
    'uploaded': uploaded,
    'time_stamp': str(datetime.now()),
    'data_list': json.dumps(data),
    }

rq = requests.post(
"https://XXXXX.pythonanywhere.com/XXX/default/api/packed_data.json",
data=data,
auth=auth,
headers={'Connection':'close'}
)

Вот (стандартный) код Web2py на стороне сервера:

def api():  
    response.view = 'generic.'+request.extension
    def GET(*args,**vars):
        patterns = 'auto'
        parser = db.parse_as_rest(patterns,args,vars)
        if parser.status == 200:
            return dict(content=parser.response)
        else:
            raise HTTP(parser.status,parser.error)
    def POST(table_name,**vars):
        return db[table_name].validate_and_insert(**vars)
    def PUT(table_name,record_id,**vars):
        return db(db[table_name]._id==record_id).update(**vars)
    def DELETE(table_name,record_id):
        return db(db[table_name]._id==record_id).delete()
    return dict(GET=GET, POST=POST, PUT=PUT, DELETE=DELETE)

Ошибка возвращает «Response 200», нов базу данных ничего не записано.

Ответы [ 2 ]

0 голосов
/ 23 сентября 2019

Контроллер возвращает вывод db[table_name].validate_and_insert(**vars)..validate_and_insert сначала запускает валидаторы, определенные для каждого поля, а затем выполняет вставку, только если нет ошибок валидации.Если запись не вставлена, это означает, что имеются ошибки проверки.

Обратите внимание, что ошибки проверки в этом случае автоматически не приводят к исключению или HTTP-ответу не-200.Вместо этого контроллер просто возвращает вывод .validate_and_insert (преобразованный в JSON), который имеет такую ​​структуру, как:

{
  id: [ID of inserted record if successful or null otherwise],
  errors: {
    'field1': 'error message 1',
    'field2': 'error message 2',
    ...
  }
}

Если вы не хотите получать ответ 200 при наличии ошибок проверки, вы можете изменитькод контроллера примерно такой:

    def POST(table_name,**vars):
        result = db[table_name].validate_and_insert(**vars)
        if result.errors:
            raise HTTP(422, 'INVALID INPUT')
        return result

В качестве альтернативы вы можете вернуть результат с ответом 200, и ваш клиентский код проверяет возвращенный JSON, чтобы определить, были ли какие-либо ошибки проверки и принять соответствующие меры.

В любом случае вы должны быть готовы обрабатывать ошибки проверки при их возникновении.

0 голосов
/ 22 сентября 2019

Я нашел ответ.Там не было ничего плохого с кодом.Web2Py устанавливает длину по умолчанию для строковых полей.Определение длины поля в db.py для большого числа решило проблему:

db.define_table('packed_data',
                Field('unitid', 'string'),
                Field('uploaded', 'string'),
                Field('time_stamp', 'string'),
                Field('data_list', type='string', length=5000),
                redefine=True)
...