Bullgill Coder Ответов: 3

Код не читает JSON из запроса curl, но работает только с JSON


Привет,
Я написал API на c#, который обновляет таблицу SQL. API передает JSON в код. Если я запускаю API через Postman, код работает правильно и обновляет таблицу SQL, но если я запускаю команду CURL, я получаю сообщение об ошибке "ссылка на объект не установлена на экземпляр объекта".

Пытаясь понять, что происходит, я вижу, что значения в модели, по-видимому, не заполняются при запуске через CURL.

Это, вероятно, что-то простое, я был бы признателен, если бы кто-то мог указать мне правильное направление.

Что я уже пробовал:

Моя модель...

    public class Balance
    {
        public string customer { get; set; }
        public string amount { get; set; }
        public string type { get; set; }
}
    public class CreateUpdate : Balance
    {
    }


и раздел PUT API, который работает, если я бегу через Postman...

// PUT api/<controller>/5
        public string Put([FromBody]CreateUpdate value)
        {
            _conPU = new SqlConnection(ConfigurationManager.ConnectionStrings["TestConnection"].ConnectionString);


            var query = "insert into L_UPD (ACUID, AMNT, DATE ,COMMENTS) " +
                "values (@ACUID, CASE WHEN @TYPE = 'debit' THEN @POINTS * -1 ELSE @POINTS END, getdate() , CASE WHEN @TYPE = 'debit' THEN 'REDEEM : ONLINE' ELSE 'EARN : ONLINE' END)";
            SqlCommand insertcommand = new SqlCommand(query, _conPU);
            insertcommand.Parameters.AddWithValue("@ACUID", value.customer);
            insertcommand.Parameters.AddWithValue("@POINTS", value.amount);
            insertcommand.Parameters.AddWithValue("@TYPE", value.type);
            _conPU.Open();
            int result = insertcommand.ExecuteNonQuery();
            if (result > 0)
            {
                return "approved";
            }
            else
            {
                return "update failed";
            }
        }


Запрос curl-это
curl -X PUT -H "Content-Type: application/json" -d '{"customer":"435673","amount":5,"type":"debit"}' http://localhost:55634/api/balance


РЕДАКТИРОВАТЬ :
Через почтальона, если я попробую просто json, например
{"customer":"435673","amount":5,"type":"debit"}

код работает, но в postman, если я использую curl
curl -X PUT -H "Content-Type: application/json" -d '{"customer":"435673","amount":5,"type":"debit"}'http://localhost:55634/api/balance

это не работает.

F-ES Sitecore

Вы можете заставить postman сгенерировать команду curl для вас, нажав на ссылку "код" и выбрав cURL из опций. Это работает? Если это так, измените его, удалив вещи, которые не появляются в вашем, пока он не перестанет работать, и то, что вы удалили последним, вероятно, даст вам знать, в чем была проблема.

Bullgill Coder

Я сгенерировал команду curl и попробовал выполнить запрос выше в postman и получил ту же ошибку для обоих. Проблема в том, что мой код - при использовании curl, похоже, не передает JSON в модель.

F-ES Sitecore

Что - то, что нужно попробовать, - это использовать HttpRequestMessage вместо FromBody

https://stackoverflow.com/questions/39030314/webapi-frombody-always-null

посмотрите, если это, по крайней мере, видя ваш json как строку, если это так, вы можете просто десериализовать эту строку самостоятельно в свой класс CreateUpdate.

Bullgill Coder

Когда я перехожу на HttpRequestMessage (если я делаю это правильно) Значение (значение CreateUpdate) имеет значение null, когда я использовать команду curl, но опять же, когда я просто использовать JSON значение заполняется правильно.

Sandeep Mewara

http://localhost:55634/api/balance => является ли этот URL-адрес действительным? Спрашивая об этом, основываясь на порте.

Это доступно?

Кроме того, поверьте, что ваш запрос здесь больше о том, почему CURL не работает, а затем ошибка ссылки на объект. Вы знаете, почему у вас есть эта ошибка (так как ответ не находится в желаемом формате, и вы не обработали этот плохой ответ)

Richard Deeming

Если вы используете curl на Windows, есть предложение в этот поток переполнения стека[^] что размещение одинарных кавычек вокруг JSON не работает.

3 Ответов

Рейтинг:
12

Bullgill Coder

Мне удалось заставить все работать, изменив свой код на HttpRequestMessage

[HttpPut]
public void Confirmation(HttpRequestMessage request)
{
    var content = request.Content;
    _conPU = new SqlConnection(ConfigurationManager.ConnectionStrings["Test_Connection"].ConnectionString);
    string jsonContent = content.ReadAsStringAsync().Result;
    jsonContent = jsonContent.Replace("'{", "{'").Replace(":", "':'").Replace(",", "','").Replace("}'", "'}");
    CreateUpdate cpu = JsonConvert.DeserializeObject<CreatePointsUpdate>(jsonContent);

     var query = "insert into L_UPD (ACUID, AMNT, DATE ,COMMENTS) " +
        "values (@ACUID, CASE WHEN @TYPE = 'debit' THEN @POINTS * -1 ELSE @POINTS END, getdate() , CASE WHEN @TYPE = 'debit' THEN 'REDEEM : ONLINE' ELSE 'EARN : ONLINE' END)";

     SqlCommand insertcommand = new SqlCommand(query, _conPU);
     insertcommand.Parameters.AddWithValue("@ACUID", cpu.customer_urn, null);
     insertcommand.Parameters.AddWithValue("@POINTS", cpu.amount, null);
     insertcommand.Parameters.AddWithValue("@TYPE", cpu.type, null);
     _conPU.Open();

     int result = insertcommand.ExecuteNonQuery();

    return;
}


Рейтинг:
1

OriginalGriff

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

Позвольте мне просто объяснить, что означает ошибка: вы пытались использовать переменную, свойство или возвращаемое значение метода, но оно содержит null - что означает, что в переменной нет экземпляра класса.
Это немного похоже на карман: у вас есть карман в рубашке, который вы используете, чтобы держать ручку. Если вы сунете руку в карман и обнаружите, что там нет ручки, вы не сможете подписать свое имя на листе бумаги - и вы получите очень смешные взгляды, если попытаетесь! Пустой карман дает вам нулевое значение (здесь нет ручки!), поэтому вы не можете сделать ничего, что обычно делали бы, когда вы извлекли свою ручку. Почему он пуст? Вот в чем вопрос - может быть, вы забыли взять ручку, когда уходили из дома сегодня утром, или, возможно, вы оставили ручку в кармане вчерашней рубашки, когда снимали ее вчера вечером.

Мы не можем сказать, потому что нас там не было, и, что еще важнее, мы даже не можем видеть вашу рубашку, не говоря уже о том, что находится в кармане!

Вернемся к компьютерам, и вы каким - то образом сделали то же самое-и мы не можем увидеть ваш код, а тем более запустить его и узнать, что содержит null, когда это не должно быть.
Но вы можете - и Visual Studio поможет вам здесь. Запустите свою программу в отладчике, и когда она выйдет из строя, VS покажет вам строку, в которой она обнаружила проблему. Затем вы можете начать просматривать различные его части, чтобы увидеть, какое значение равно null, и начать просматривать свой код, чтобы узнать, почему. Поэтому поставьте точку останова в начале метода, содержащего строку ошибки, и снова запустите программу с самого начала. На этот раз VS остановится перед ошибкой и позволит вам изучить, что происходит, пройдя через код, глядя на ваши значения.

Но мы не можем этого сделать - у нас нет вашего кода, мы не знаем, как его использовать, если бы он у нас был, у нас нет ваших данных. Так что попробуйте - и посмотрите, сколько информации вы можете узнать!
Я бы начал с того, чтобы посмотреть, что ConfigurationManager.ConnectionStrings["TestConnection"] возвращается. Если это так ... null вы немедленно получите сообщение об ошибке.


Bullgill Coder

Как уже объяснялось, я знаю, что в кармане моей рубашки есть ручка, так как когда я просто отправляю json из команды curl, все работает правильно, и мои данные datatable заполняются. Проблема, похоже, заключается в том, что если я оберну json в команду curl, то не смогу найти ручку в кармане, хотя знаю, что она все еще там!
Из моего кода (выше) проблема заключается в том, что
public string Put([FromBody]CreateUpdate value)
при использовании json заполняется только значение, а при использовании curl-нет.

Рейтинг:
1

Dave Kreskowiak

Ну, первое, что я вижу с вашим контроллером, - это то, что он предполагает, что "значение", переданное в метод, не является нулевым и содержит допустимые данные. Вы вообще не проверяете "ценность".

..и это приводит к ошибке, которую вы получаете. В Почтальона, это обеспечивает данные, и правильно отформатирован в формат JSON, метод АПИ. Ваша команда CURL выполняет тот же вызов, но не предоставляет данные правильно, поэтому вызов завершается неудачей.


Bullgill Coder

Я планирую добавить проверку, как только все заработает. Шагая по коду, я вижу, что класс "value" заполняется не curl, а json.

Dave Kreskowiak

Это была плохая идея. Нет, вам нужно добавить код проверки и начать тестирование заново, а также добавить еще одно тестирование для пограничных случаев.

Ты не стал экономить время, а удвоил его. Добавление проверки сказало бы вам, что есть проблема с передаваемыми параметрами, и сосредоточило бы ваши усилия по устранению неполадок на тестовом инструменте, а не на вашем коде.