blob: da8b23369d2946770a784d0c22cfd52dafad8077 [file] [log] [blame]
Matt Spinleracb7c102020-01-10 13:49:22 -06001/**
2 * Copyright © 2020 IBM Corporation
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "user_data_json.hpp"
18
Harisuddin Mohamed Isabebeb942020-03-12 17:12:24 +080019#include "json_utils.hpp"
Matt Spinleracb7c102020-01-10 13:49:22 -060020#include "pel_types.hpp"
Harisuddin Mohamed Isabebeb942020-03-12 17:12:24 +080021#include "pel_values.hpp"
Matt Spinler18207142020-03-26 14:45:54 -050022#include "stream.hpp"
Matt Spinleracb7c102020-01-10 13:49:22 -060023#include "user_data_formats.hpp"
24
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +080025#include <Python.h>
26
Matt Spinleracb7c102020-01-10 13:49:22 -060027#include <iomanip>
28#include <nlohmann/json.hpp>
29#include <phosphor-logging/log.hpp>
30#include <sstream>
31
32namespace openpower::pels::user_data
33{
Harisuddin Mohamed Isabebeb942020-03-12 17:12:24 +080034namespace pv = openpower::pels::pel_values;
Matt Spinleracb7c102020-01-10 13:49:22 -060035using namespace phosphor::logging;
Sumit Kumar516935a2021-04-14 13:00:54 -050036using orderedJSON = nlohmann::ordered_json;
Matt Spinleracb7c102020-01-10 13:49:22 -060037
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +080038void pyDecRef(PyObject* pyObj)
39{
40 Py_XDECREF(pyObj);
41}
42
Matt Spinleracb7c102020-01-10 13:49:22 -060043/**
44 * @brief Returns a JSON string for use by PEL::printSectionInJSON().
45 *
46 * The returning string will contain a JSON object, but without
47 * the outer {}. If the input JSON isn't a JSON object (dict), then
48 * one will be created with the input added to a 'Data' key.
49 *
50 * @param[in] json - The JSON to convert to a string
51 *
52 * @return std::string - The JSON string
53 */
54std::string prettyJSON(uint16_t componentID, uint8_t subType, uint8_t version,
Sumit Kumar516935a2021-04-14 13:00:54 -050055 const orderedJSON& json)
Matt Spinleracb7c102020-01-10 13:49:22 -060056{
Sumit Kumar516935a2021-04-14 13:00:54 -050057 orderedJSON output;
Harisuddin Mohamed Isabebeb942020-03-12 17:12:24 +080058 output[pv::sectionVer] = std::to_string(version);
59 output[pv::subSection] = std::to_string(subType);
60 output[pv::createdBy] = getNumberString("0x%04X", componentID);
Matt Spinleracb7c102020-01-10 13:49:22 -060061
62 if (!json.is_object())
63 {
64 output["Data"] = json;
65 }
66 else
67 {
68 for (const auto& [key, value] : json.items())
69 {
70 output[key] = value;
71 }
72 }
73
74 // Let nlohmann do the pretty printing.
75 std::stringstream stream;
76 stream << std::setw(4) << output;
77
78 auto jsonString = stream.str();
79
80 // Now it looks like:
81 // {
82 // "Section Version": ...
83 // ...
84 // }
85
86 // Since PEL::printSectionInJSON() will supply the outer { }s,
87 // remove the existing ones.
88
89 // Replace the { and the following newline, and the } and its
90 // preceeding newline.
91 jsonString.erase(0, 2);
92
93 auto pos = jsonString.find_last_of('}');
94 jsonString.erase(pos - 1);
95
96 return jsonString;
97}
98
99/**
Matt Spinler18207142020-03-26 14:45:54 -0500100 * @brief Return a JSON string from the passed in CBOR data.
101 *
102 * @param[in] componentID - The comp ID from the UserData section header
103 * @param[in] subType - The subtype from the UserData section header
104 * @param[in] version - The version from the UserData section header
105 * @param[in] data - The CBOR data
106 *
107 * @return std::string - The JSON string
108 */
109std::string getCBORJSON(uint16_t componentID, uint8_t subType, uint8_t version,
110 const std::vector<uint8_t>& data)
111{
112 // The CBOR parser needs the pad bytes added to 4 byte align
113 // removed. The number of bytes added to the pad is on the
114 // very end, so will remove both fields before parsing.
115
116 // If the data vector is too short, an exception will get
117 // thrown which will be handled up the call stack.
118
119 auto cborData = data;
120 uint32_t pad{};
121
122 Stream stream{cborData};
123 stream.offset(cborData.size() - 4);
124 stream >> pad;
125
126 if (cborData.size() > (pad + sizeof(pad)))
127 {
128 cborData.resize(data.size() - sizeof(pad) - pad);
129 }
130
Matt Spinlerbb1c1d52021-06-03 13:18:48 -0600131 orderedJSON json = orderedJSON::from_cbor(cborData);
Matt Spinler18207142020-03-26 14:45:54 -0500132
133 return prettyJSON(componentID, subType, version, json);
134}
135
136/**
137 * @brief Return a JSON string from the passed in text data.
138 *
Harisuddin Mohamed Isaaadf28f2020-09-29 22:10:52 +0800139 * The function breaks up the input text into a vector of strings with
140 * newline as separator and converts that into JSON. It will convert any
141 * unprintable characters to periods.
Matt Spinler18207142020-03-26 14:45:54 -0500142 *
143 * @param[in] componentID - The comp ID from the UserData section header
144 * @param[in] subType - The subtype from the UserData section header
145 * @param[in] version - The version from the UserData section header
146 * @param[in] data - The CBOR data
147 *
148 * @return std::string - The JSON string
149 */
150std::string getTextJSON(uint16_t componentID, uint8_t subType, uint8_t version,
151 const std::vector<uint8_t>& data)
152{
Matt Spinler18207142020-03-26 14:45:54 -0500153 std::vector<std::string> text;
154 size_t startPos = 0;
Matt Spinler18207142020-03-26 14:45:54 -0500155
Harisuddin Mohamed Isaaadf28f2020-09-29 22:10:52 +0800156 // Converts any unprintable characters to periods
Matt Spinler18207142020-03-26 14:45:54 -0500157 auto validate = [](char& ch) {
158 if ((ch < ' ') || (ch > '~'))
159 {
160 ch = '.';
161 }
162 };
163
Harisuddin Mohamed Isaaadf28f2020-09-29 22:10:52 +0800164 // Break up the data into an array of strings with newline as separator
165 for (size_t pos = 0; pos < data.size(); ++pos)
Matt Spinler18207142020-03-26 14:45:54 -0500166 {
Harisuddin Mohamed Isaaadf28f2020-09-29 22:10:52 +0800167 if (data[pos] == '\n')
Matt Spinler18207142020-03-26 14:45:54 -0500168 {
169 std::string line{reinterpret_cast<const char*>(&data[startPos]),
Harisuddin Mohamed Isaaadf28f2020-09-29 22:10:52 +0800170 pos - startPos};
Matt Spinler18207142020-03-26 14:45:54 -0500171 std::for_each(line.begin(), line.end(), validate);
172 text.push_back(std::move(line));
Harisuddin Mohamed Isaaadf28f2020-09-29 22:10:52 +0800173 startPos = pos + 1;
Matt Spinler18207142020-03-26 14:45:54 -0500174 }
Harisuddin Mohamed Isaaadf28f2020-09-29 22:10:52 +0800175 }
176 if (startPos < data.size())
177 {
178 std::string line{reinterpret_cast<const char*>(&data[startPos]),
179 data.size() - startPos};
180 std::for_each(line.begin(), line.end(), validate);
181 text.push_back(std::move(line));
Matt Spinler18207142020-03-26 14:45:54 -0500182 }
183
Sumit Kumar516935a2021-04-14 13:00:54 -0500184 orderedJSON json = text;
Matt Spinler18207142020-03-26 14:45:54 -0500185 return prettyJSON(componentID, subType, version, json);
186}
187
188/**
Matt Spinleracb7c102020-01-10 13:49:22 -0600189 * @brief Convert to an appropriate JSON string as the data is one of
190 * the formats that we natively support.
191 *
192 * @param[in] componentID - The comp ID from the UserData section header
193 * @param[in] subType - The subtype from the UserData section header
194 * @param[in] version - The version from the UserData section header
195 * @param[in] data - The data itself
196 *
197 * @return std::optional<std::string> - The JSON string if it could be created,
198 * else std::nullopt.
199 */
200std::optional<std::string>
201 getBuiltinFormatJSON(uint16_t componentID, uint8_t subType, uint8_t version,
202 const std::vector<uint8_t>& data)
203{
204 switch (subType)
205 {
206 case static_cast<uint8_t>(UserDataFormat::json):
207 {
208 std::string jsonString{data.begin(), data.begin() + data.size()};
209
Matt Spinlerbb1c1d52021-06-03 13:18:48 -0600210 orderedJSON json = orderedJSON::parse(jsonString);
Matt Spinleracb7c102020-01-10 13:49:22 -0600211
212 return prettyJSON(componentID, subType, version, json);
213 }
Matt Spinler18207142020-03-26 14:45:54 -0500214 case static_cast<uint8_t>(UserDataFormat::cbor):
215 {
216 return getCBORJSON(componentID, subType, version, data);
217 }
218 case static_cast<uint8_t>(UserDataFormat::text):
219 {
220 return getTextJSON(componentID, subType, version, data);
221 }
Matt Spinleracb7c102020-01-10 13:49:22 -0600222 default:
223 break;
224 }
225 return std::nullopt;
226}
227
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800228/**
229 * @brief Call Python modules to parse the data into a JSON string
230 *
231 * The module to call is based on the Creator Subsystem ID and the Component
232 * ID under the namespace "udparsers". For example: "udparsers.xyyyy.xyyyy"
233 * where "x" is the Creator Subsystem ID and "yyyy" is the Component ID.
234 *
235 * All modules must provide the following:
236 * Function: parseUDToJson
237 * Argument list:
238 * 1. (int) Sub-section type
239 * 2. (int) Section version
240 * 3. (memoryview): Data
241 *-Return data:
242 * 1. (str) JSON string
243 *
244 * @param[in] componentID - The comp ID from the UserData section header
245 * @param[in] subType - The subtype from the UserData section header
246 * @param[in] version - The version from the UserData section header
247 * @param[in] data - The data itself
248 * @param[in] creatorID - The creatorID from the PrivateHeader section
249 * @return std::optional<std::string> - The JSON string if it could be created,
250 * else std::nullopt
251 */
252std::optional<std::string> getPythonJSON(uint16_t componentID, uint8_t subType,
253 uint8_t version,
254 const std::vector<uint8_t>& data,
255 uint8_t creatorID)
256{
Matt Spinlerbe952d22022-07-01 11:30:11 -0500257 PyObject *pName, *pModule, *eType, *eValue, *eTraceback, *pKey;
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800258 std::string pErrStr;
259 std::string module = getNumberString("%c", tolower(creatorID)) +
260 getNumberString("%04x", componentID);
261 pName = PyUnicode_FromString(
262 std::string("udparsers." + module + "." + module).c_str());
263 std::unique_ptr<PyObject, decltype(&pyDecRef)> modNamePtr(pName, &pyDecRef);
264 pModule = PyImport_Import(pName);
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800265 if (pModule == NULL)
266 {
267 pErrStr = "No error string found";
268 PyErr_Fetch(&eType, &eValue, &eTraceback);
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800269 if (eType)
270 {
271 Py_XDECREF(eType);
272 }
273 if (eTraceback)
274 {
275 Py_XDECREF(eTraceback);
276 }
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800277 if (eValue)
278 {
279 PyObject* pStr = PyObject_Str(eValue);
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800280 Py_XDECREF(eValue);
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800281 if (pStr)
282 {
283 pErrStr = PyUnicode_AsUTF8(pStr);
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800284 Py_XDECREF(pStr);
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800285 }
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800286 }
287 }
288 else
289 {
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800290 std::unique_ptr<PyObject, decltype(&pyDecRef)> modPtr(pModule,
291 &pyDecRef);
292 std::string funcToCall = "parseUDToJson";
293 pKey = PyUnicode_FromString(funcToCall.c_str());
294 std::unique_ptr<PyObject, decltype(&pyDecRef)> keyPtr(pKey, &pyDecRef);
Matt Spinlerbe952d22022-07-01 11:30:11 -0500295 PyObject* pDict = PyModule_GetDict(pModule);
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800296 Py_INCREF(pDict);
297 if (!PyDict_Contains(pDict, pKey))
298 {
299 Py_DECREF(pDict);
300 log<level::ERR>(
301 "Python module error",
302 entry("ERROR=%s",
303 std::string(funcToCall + " function missing").c_str()),
304 entry("PARSER_MODULE=%s", module.c_str()),
305 entry("SUBTYPE=0x%X", subType), entry("VERSION=%d", version),
306 entry("DATA_LENGTH=%lu\n", data.size()));
307 return std::nullopt;
308 }
Matt Spinlerbe952d22022-07-01 11:30:11 -0500309 PyObject* pFunc = PyDict_GetItemString(pDict, funcToCall.c_str());
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800310 Py_DECREF(pDict);
311 Py_INCREF(pFunc);
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800312 if (PyCallable_Check(pFunc))
313 {
314 auto ud = data.data();
Matt Spinlerbe952d22022-07-01 11:30:11 -0500315 PyObject* pArgs = PyTuple_New(3);
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800316 std::unique_ptr<PyObject, decltype(&pyDecRef)> argPtr(pArgs,
317 &pyDecRef);
318 PyTuple_SetItem(pArgs, 0,
319 PyLong_FromUnsignedLong((unsigned long)subType));
320 PyTuple_SetItem(pArgs, 1,
321 PyLong_FromUnsignedLong((unsigned long)version));
Matt Spinlerbe952d22022-07-01 11:30:11 -0500322 PyObject* pData = PyMemoryView_FromMemory(
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800323 reinterpret_cast<char*>(const_cast<unsigned char*>(ud)),
324 data.size(), PyBUF_READ);
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800325 PyTuple_SetItem(pArgs, 2, pData);
Matt Spinlerbe952d22022-07-01 11:30:11 -0500326 PyObject* pResult = PyObject_CallObject(pFunc, pArgs);
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800327 Py_DECREF(pFunc);
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800328 if (pResult)
329 {
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800330 std::unique_ptr<PyObject, decltype(&pyDecRef)> resPtr(
331 pResult, &pyDecRef);
Matt Spinlerbe952d22022-07-01 11:30:11 -0500332 PyObject* pBytes =
333 PyUnicode_AsEncodedString(pResult, "utf-8", "~E~");
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800334 std::unique_ptr<PyObject, decltype(&pyDecRef)> pyBytePtr(
335 pBytes, &pyDecRef);
336 const char* output = PyBytes_AS_STRING(pBytes);
337 try
338 {
Matt Spinlerbb1c1d52021-06-03 13:18:48 -0600339 orderedJSON json = orderedJSON::parse(output);
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800340 if ((json.is_object() && !json.empty()) ||
341 (json.is_array() && json.size() > 0) ||
342 (json.is_string() && json != ""))
343 {
344 return prettyJSON(componentID, subType, version, json);
345 }
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800346 }
Patrick Williams66491c62021-10-06 12:23:37 -0500347 catch (const std::exception& e)
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800348 {
349 log<level::ERR>("Bad JSON from parser",
350 entry("ERROR=%s", e.what()),
351 entry("PARSER_MODULE=%s", module.c_str()),
352 entry("SUBTYPE=0x%X", subType),
353 entry("VERSION=%d", version),
354 entry("DATA_LENGTH=%lu\n", data.size()));
355 return std::nullopt;
356 }
357 }
358 else
359 {
360 pErrStr = "No error string found";
361 PyErr_Fetch(&eType, &eValue, &eTraceback);
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800362 if (eType)
363 {
364 Py_XDECREF(eType);
365 }
366 if (eTraceback)
367 {
368 Py_XDECREF(eTraceback);
369 }
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800370 if (eValue)
371 {
372 PyObject* pStr = PyObject_Str(eValue);
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800373 Py_XDECREF(eValue);
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800374 if (pStr)
375 {
376 pErrStr = PyUnicode_AsUTF8(pStr);
Harisuddin Mohamed Isad5c31362021-05-29 13:33:39 +0800377 Py_XDECREF(pStr);
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800378 }
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800379 }
380 }
381 }
382 }
383 if (!pErrStr.empty())
384 {
Matt Spinler3279cc52022-02-15 11:02:00 -0600385 log<level::DEBUG>("Python exception thrown by parser",
386 entry("ERROR=%s", pErrStr.c_str()),
387 entry("PARSER_MODULE=%s", module.c_str()),
388 entry("SUBTYPE=0x%X", subType),
389 entry("VERSION=%d", version),
390 entry("DATA_LENGTH=%lu\n", data.size()));
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800391 }
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800392 return std::nullopt;
393}
394
Matt Spinleracb7c102020-01-10 13:49:22 -0600395std::optional<std::string> getJSON(uint16_t componentID, uint8_t subType,
396 uint8_t version,
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800397 const std::vector<uint8_t>& data,
Harisuddin Mohamed Isa3fdcd4e2020-08-26 11:56:42 +0800398 uint8_t creatorID,
399 const std::vector<std::string>& plugins)
Matt Spinleracb7c102020-01-10 13:49:22 -0600400{
Harisuddin Mohamed Isa3fdcd4e2020-08-26 11:56:42 +0800401 std::string subsystem = getNumberString("%c", tolower(creatorID));
402 std::string component = getNumberString("%04x", componentID);
Matt Spinleracb7c102020-01-10 13:49:22 -0600403 try
404 {
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800405 if (pv::creatorIDs.at(getNumberString("%c", creatorID)) == "BMC" &&
406 componentID == static_cast<uint16_t>(ComponentID::phosphorLogging))
Matt Spinleracb7c102020-01-10 13:49:22 -0600407 {
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800408 return getBuiltinFormatJSON(componentID, subType, version, data);
409 }
Harisuddin Mohamed Isa3fdcd4e2020-08-26 11:56:42 +0800410 else if (std::find(plugins.begin(), plugins.end(),
411 subsystem + component) != plugins.end())
Harisuddin Mohamed Isaf67bafd2020-07-06 17:51:21 +0800412 {
413 return getPythonJSON(componentID, subType, version, data,
414 creatorID);
Matt Spinleracb7c102020-01-10 13:49:22 -0600415 }
416 }
Patrick Williams66491c62021-10-06 12:23:37 -0500417 catch (const std::exception& e)
Matt Spinleracb7c102020-01-10 13:49:22 -0600418 {
Matt Spinler18207142020-03-26 14:45:54 -0500419 log<level::ERR>("Failed parsing UserData", entry("ERROR=%s", e.what()),
420 entry("COMP_ID=0x%X", componentID),
421 entry("SUBTYPE=0x%X", subType),
422 entry("VERSION=%d", version),
423 entry("DATA_LENGTH=%lu\n", data.size()));
Matt Spinleracb7c102020-01-10 13:49:22 -0600424 }
425
426 return std::nullopt;
427}
428
429} // namespace openpower::pels::user_data