1 /**
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 #pragma once
17 
18 #include "action.hpp"
19 #include "chassis.hpp"
20 #include "configuration.hpp"
21 #include "device.hpp"
22 #include "i2c_interface.hpp"
23 #include "i2c_write_bit_action.hpp"
24 #include "i2c_write_byte_action.hpp"
25 #include "i2c_write_bytes_action.hpp"
26 #include "pmbus_write_vout_command_action.hpp"
27 #include "presence_detection.hpp"
28 #include "rail.hpp"
29 #include "rule.hpp"
30 #include "run_rule_action.hpp"
31 #include "sensor_monitoring.hpp"
32 
33 #include <nlohmann/json.hpp>
34 
35 #include <cstdint>
36 #include <filesystem>
37 #include <memory>
38 #include <stdexcept>
39 #include <string>
40 #include <tuple>
41 #include <vector>
42 
43 namespace phosphor::power::regulators::config_file_parser
44 {
45 
46 /**
47  * Parses the specified JSON configuration file.
48  *
49  * Returns the corresponding C++ Rule and Chassis objects.
50  *
51  * Throws a ConfigFileParserError if an error occurs.
52  *
53  * @param pathName configuration file path name
54  * @return tuple containing vectors of Rule and Chassis objects
55  */
56 std::tuple<std::vector<std::unique_ptr<Rule>>,
57            std::vector<std::unique_ptr<Chassis>>>
58     parse(const std::filesystem::path& pathName);
59 
60 /*
61  * Internal implementation details for parse()
62  */
63 namespace internal
64 {
65 
66 /**
67  * Returns the specified property of the specified JSON element.
68  *
69  * Throws an invalid_argument exception if the property does not exist.
70  *
71  * @param element JSON element
72  * @param property property name
73  */
74 inline const nlohmann::json& getRequiredProperty(const nlohmann::json& element,
75                                                  const std::string& property)
76 {
77     auto it = element.find(property);
78     if (it == element.end())
79     {
80         throw std::invalid_argument{"Required property missing: " + property};
81     }
82     return *it;
83 }
84 
85 /**
86  * Parses a JSON element containing an action.
87  *
88  * Returns the corresponding C++ Action object.
89  *
90  * Throws an exception if parsing fails.
91  *
92  * @param element JSON element
93  * @return Action object
94  */
95 std::unique_ptr<Action> parseAction(const nlohmann::json& element);
96 
97 /**
98  * Parses a JSON element containing an array of actions.
99  *
100  * Returns the corresponding C++ Action objects.
101  *
102  * Throws an exception if parsing fails.
103  *
104  * @param element JSON element
105  * @return vector of Action objects
106  */
107 std::vector<std::unique_ptr<Action>>
108     parseActionArray(const nlohmann::json& element);
109 
110 /**
111  * Parses a JSON element containing a bit position (from 0-7).
112  *
113  * Returns the corresponding C++ uint8_t value.
114  *
115  * Throws an exception if parsing fails.
116  *
117  * @param element JSON element
118  * @return uint8_t value
119  */
120 inline uint8_t parseBitPosition(const nlohmann::json& element)
121 {
122     // Verify element contains an integer
123     if (!element.is_number_integer())
124     {
125         throw std::invalid_argument{"Element is not an integer"};
126     }
127     int value = element.get<int>();
128     if ((value < 0) || (value > 7))
129     {
130         throw std::invalid_argument{"Element is not a bit position"};
131     }
132     return static_cast<uint8_t>(value);
133 }
134 
135 /**
136  * Parses a JSON element containing a bit value (0 or 1).
137  *
138  * Returns the corresponding C++ uint8_t value.
139  *
140  * Throws an exception if parsing fails.
141  *
142  * @param element JSON element
143  * @return uint8_t value
144  */
145 inline uint8_t parseBitValue(const nlohmann::json& element)
146 {
147     // Verify element contains an integer
148     if (!element.is_number_integer())
149     {
150         throw std::invalid_argument{"Element is not an integer"};
151     }
152     int value = element.get<int>();
153     if ((value < 0) || (value > 1))
154     {
155         throw std::invalid_argument{"Element is not a bit value"};
156     }
157     return static_cast<uint8_t>(value);
158 }
159 
160 /**
161  * Parses a JSON element containing a boolean.
162  *
163  * Returns the corresponding C++ boolean value.
164  *
165  * Throws an exception if parsing fails.
166  *
167  * @param element JSON element
168  * @return boolean value
169  */
170 inline bool parseBoolean(const nlohmann::json& element)
171 {
172     // Verify element contains a boolean
173     if (!element.is_boolean())
174     {
175         throw std::invalid_argument{"Element is not a boolean"};
176     }
177     return element.get<bool>();
178 }
179 
180 /**
181  * Parses a JSON element containing a chassis.
182  *
183  * Returns the corresponding C++ Chassis object.
184  *
185  * Throws an exception if parsing fails.
186  *
187  * @param element JSON element
188  * @return Chassis object
189  */
190 std::unique_ptr<Chassis> parseChassis(const nlohmann::json& element);
191 
192 /**
193  * Parses a JSON element containing an array of chassis.
194  *
195  * Returns the corresponding C++ Chassis objects.
196  *
197  * Throws an exception if parsing fails.
198  *
199  * @param element JSON element
200  * @return vector of Chassis objects
201  */
202 std::vector<std::unique_ptr<Chassis>>
203     parseChassisArray(const nlohmann::json& element);
204 
205 /**
206  * Parses a JSON element containing a configuration.
207  *
208  * Returns the corresponding C++ Configuration object.
209  *
210  * Throws an exception if parsing fails.
211  *
212  * @param element JSON element
213  * @return Configuration object
214  */
215 std::unique_ptr<Configuration>
216     parseConfiguration(const nlohmann::json& element);
217 
218 /**
219  * Parses a JSON element containing a device.
220  *
221  * Returns the corresponding C++ Device object.
222  *
223  * Throws an exception if parsing fails.
224  *
225  * @param element JSON element
226  * @return Device object
227  */
228 std::unique_ptr<Device> parseDevice(const nlohmann::json& element);
229 
230 /**
231  * Parses a JSON element containing an array of devices.
232  *
233  * Returns the corresponding C++ Device objects.
234  *
235  * Throws an exception if parsing fails.
236  *
237  * @param element JSON element
238  * @return vector of Device objects
239  */
240 std::vector<std::unique_ptr<Device>>
241     parseDeviceArray(const nlohmann::json& element);
242 
243 /**
244  * Parses a JSON element containing a double (floating point number).
245  *
246  * Returns the corresponding C++ double value.
247  *
248  * Throws an exception if parsing fails.
249  *
250  * @param element JSON element
251  * @return double value
252  */
253 inline double parseDouble(const nlohmann::json& element)
254 {
255     // Verify element contains a number (integer or floating point)
256     if (!element.is_number())
257     {
258         throw std::invalid_argument{"Element is not a number"};
259     }
260     return element.get<double>();
261 }
262 
263 /**
264  * Parses a JSON element containing a byte value expressed as a hexadecimal
265  * string.
266  *
267  * The JSON number data type does not support the hexadecimal format.  For this
268  * reason, hexadecimal byte values are stored as strings in the configuration
269  * file.
270  *
271  * Returns the corresponding C++ uint8_t value.
272  *
273  * Throws an exception if parsing fails.
274  *
275  * @param element JSON element
276  * @return uint8_t value
277  */
278 inline uint8_t parseHexByte(const nlohmann::json& element)
279 {
280     if (!element.is_string())
281     {
282         throw std::invalid_argument{"Element is not a string"};
283     }
284     std::string value = element.get<std::string>();
285 
286     bool isHex = (value.compare(0, 2, "0x") == 0) && (value.size() > 2) &&
287                  (value.size() < 5) &&
288                  (value.find_first_not_of("0123456789abcdefABCDEF", 2) ==
289                   std::string::npos);
290     if (!isHex)
291     {
292         throw std::invalid_argument{"Element is not hexadecimal string"};
293     }
294     return static_cast<uint8_t>(std::stoul(value, 0, 0));
295 }
296 
297 /**
298  * Parses a JSON element containing an array of byte values expressed as a
299  * hexadecimal strings.
300  *
301  * Returns the corresponding C++ uint8_t values.
302  *
303  * Throws an exception if parsing fails.
304  *
305  * @param element JSON element
306  * @return vector of uint8_t
307  */
308 std::vector<uint8_t> parseHexByteArray(const nlohmann::json& element);
309 
310 /**
311  * Parses a JSON element containing an i2c_interface.
312  *
313  * Returns the corresponding C++ i2c::I2CInterface object.
314  *
315  * Throws an exception if parsing fails.
316  *
317  * @param element JSON element
318  * @return i2c::I2CInterface object
319  */
320 std::unique_ptr<i2c::I2CInterface>
321     parseI2CInterface(const nlohmann::json& element);
322 
323 /**
324  * Parses a JSON element containing an i2c_write_bit action.
325  *
326  * Returns the corresponding C++ I2CWriteBitAction object.
327  *
328  * Throws an exception if parsing fails.
329  *
330  * @param element JSON element
331  * @return I2CWriteBitAction object
332  */
333 std::unique_ptr<I2CWriteBitAction>
334     parseI2CWriteBit(const nlohmann::json& element);
335 
336 /**
337  * Parses a JSON element containing an i2c_write_byte action.
338  *
339  * Returns the corresponding C++ I2CWriteByteAction object.
340  *
341  * Throws an exception if parsing fails.
342  *
343  * @param element JSON element
344  * @return I2CWriteByteAction object
345  */
346 std::unique_ptr<I2CWriteByteAction>
347     parseI2CWriteByte(const nlohmann::json& element);
348 
349 /**
350  * Parses a JSON element containing an i2c_write_bytes action.
351  *
352  * Returns the corresponding C++ I2CWriteBytesAction object.
353  *
354  * Throws an exception if parsing fails.
355  *
356  * @param element JSON element
357  * @return I2CWriteBytesAction object
358  */
359 std::unique_ptr<I2CWriteBytesAction>
360     parseI2CWriteBytes(const nlohmann::json& element);
361 
362 /**
363  * Parses a JSON element containing an 8-bit signed integer.
364  *
365  * Returns the corresponding C++ int8_t value.
366  *
367  * Throws an exception if parsing fails.
368  *
369  * @param element JSON element
370  * @return int8_t value
371  */
372 inline int8_t parseInt8(const nlohmann::json& element)
373 {
374     // Verify element contains an integer
375     if (!element.is_number_integer())
376     {
377         throw std::invalid_argument{"Element is not an integer"};
378     }
379     int value = element.get<int>();
380     if ((value < INT8_MIN) || (value > INT8_MAX))
381     {
382         throw std::invalid_argument{"Element is not an 8-bit signed integer"};
383     }
384     return static_cast<int8_t>(value);
385 }
386 
387 /**
388  * Parses a JSON element containing a pmbus_write_vout_command action.
389  *
390  * Returns the corresponding C++ PMBusWriteVoutCommandAction object.
391  *
392  * Throws an exception if parsing fails.
393  *
394  * @param element JSON element
395  * @return PMBusWriteVoutCommandAction object
396  */
397 std::unique_ptr<PMBusWriteVoutCommandAction>
398     parsePMBusWriteVoutCommand(const nlohmann::json& element);
399 
400 /**
401  * Parses a JSON element containing a rail.
402  *
403  * Returns the corresponding C++ Rail object.
404  *
405  * Throws an exception if parsing fails.
406  *
407  * @param element JSON element
408  * @return Rail object
409  */
410 std::unique_ptr<Rail> parseRail(const nlohmann::json& element);
411 
412 /**
413  * Parses a JSON element containing an array of rails.
414  *
415  * Returns the corresponding C++ Rail objects.
416  *
417  * Throws an exception if parsing fails.
418  *
419  * @param element JSON element
420  * @return vector of Rail objects
421  */
422 std::vector<std::unique_ptr<Rail>>
423     parseRailArray(const nlohmann::json& element);
424 
425 /**
426  * Parses the JSON root element of the entire configuration file.
427  *
428  * Returns the corresponding C++ Rule and Chassis objects.
429  *
430  * Throws an exception if parsing fails.
431  *
432  * @param element JSON element
433  * @return tuple containing vectors of Rule and Chassis objects
434  */
435 std::tuple<std::vector<std::unique_ptr<Rule>>,
436            std::vector<std::unique_ptr<Chassis>>>
437     parseRoot(const nlohmann::json& element);
438 
439 /**
440  * Parses a JSON element containing a rule.
441  *
442  * Returns the corresponding C++ Rule object.
443  *
444  * Throws an exception if parsing fails.
445  *
446  * @param element JSON element
447  * @return Rule object
448  */
449 std::unique_ptr<Rule> parseRule(const nlohmann::json& element);
450 
451 /**
452  * Parses a JSON element containing an array of rules.
453  *
454  * Returns the corresponding C++ Rule objects.
455  *
456  * Throws an exception if parsing fails.
457  *
458  * @param element JSON element
459  * @return vector of Rule objects
460  */
461 std::vector<std::unique_ptr<Rule>>
462     parseRuleArray(const nlohmann::json& element);
463 
464 /**
465  * Parses the "rule_id" or "actions" property in a JSON element.
466  *
467  * The element must contain one property or the other but not both.
468  *
469  * If the element contains a "rule_id" property, the corresponding C++
470  * RunRuleAction object is returned.
471  *
472  * If the element contains an "actions" property, the corresponding C++ Action
473  * objects are returned.
474  *
475  * Throws an exception if parsing fails.
476  *
477  * @param element JSON element
478  * @return vector of Action objects
479  */
480 std::vector<std::unique_ptr<Action>>
481     parseRuleIDOrActionsProperty(const nlohmann::json& element);
482 
483 /**
484  * Parses a JSON element containing a run_rule action.
485  *
486  * Returns the corresponding C++ RunRuleAction object.
487  *
488  * Throws an exception if parsing fails.
489  *
490  * @param element JSON element
491  * @return RunRuleAction object
492  */
493 std::unique_ptr<RunRuleAction> parseRunRule(const nlohmann::json& element);
494 
495 /**
496  * Parses a JSON element containing a sensor monitoring operation.
497  *
498  * Returns the corresponding C++ SensorMonitoring object.
499  *
500  * Throws an exception if parsing fails.
501  *
502  * @param element JSON element
503  * @return SensorMonitoring object
504  */
505 std::unique_ptr<SensorMonitoring>
506     parseSensorMonitoring(const nlohmann::json& element);
507 
508 /**
509  * Parses a JSON element containing a string.
510  *
511  * Returns the corresponding C++ string.
512  *
513  * Throws an exception if parsing fails.
514  *
515  * @param element JSON element
516  * @param isEmptyValid indicates whether an empty string value is valid
517  * @return string value
518  */
519 inline std::string parseString(const nlohmann::json& element,
520                                bool isEmptyValid = false)
521 {
522     if (!element.is_string())
523     {
524         throw std::invalid_argument{"Element is not a string"};
525     }
526     std::string value = element.get<std::string>();
527     if (value.empty() && !isEmptyValid)
528     {
529         throw std::invalid_argument{"Element contains an empty string"};
530     }
531     return value;
532 }
533 
534 /**
535  * Parses a JSON element containing an 8-bit unsigned integer.
536  *
537  * Returns the corresponding C++ uint8_t value.
538  *
539  * Throws an exception if parsing fails.
540  *
541  * @param element JSON element
542  * @return uint8_t value
543  */
544 inline uint8_t parseUint8(const nlohmann::json& element)
545 {
546     // Verify element contains an integer
547     if (!element.is_number_integer())
548     {
549         throw std::invalid_argument{"Element is not an integer"};
550     }
551     int value = element.get<int>();
552     if ((value < 0) || (value > UINT8_MAX))
553     {
554         throw std::invalid_argument{"Element is not an 8-bit unsigned integer"};
555     }
556     return static_cast<uint8_t>(value);
557 }
558 
559 /**
560  * Parses a JSON element containing an unsigned integer.
561  *
562  * Returns the corresponding C++ unsigned int value.
563  *
564  * Throws an exception if parsing fails.
565  *
566  * @param element JSON element
567  * @return unsigned int value
568  */
569 inline unsigned int parseUnsignedInteger(const nlohmann::json& element)
570 {
571     // Verify element contains an unsigned integer
572     if (!element.is_number_unsigned())
573     {
574         throw std::invalid_argument{"Element is not an unsigned integer"};
575     }
576     return element.get<unsigned int>();
577 }
578 
579 /**
580  * Verifies that the specified JSON element is a JSON array.
581  *
582  * Throws an invalid_argument exception if the element is not an array.
583  *
584  * @param element JSON element
585  */
586 inline void verifyIsArray(const nlohmann::json& element)
587 {
588     if (!element.is_array())
589     {
590         throw std::invalid_argument{"Element is not an array"};
591     }
592 }
593 
594 /**
595  * Verifies that the specified JSON element is a JSON object.
596  *
597  * Throws an invalid_argument exception if the element is not an object.
598  *
599  * @param element JSON element
600  */
601 inline void verifyIsObject(const nlohmann::json& element)
602 {
603     if (!element.is_object())
604     {
605         throw std::invalid_argument{"Element is not an object"};
606     }
607 }
608 
609 /**
610  * Verifies that the specified JSON element contains the expected number of
611  * properties.
612  *
613  * Throws an invalid_argument exception if the element contains a different
614  * number of properties.  This indicates the element contains an invalid
615  * property.
616  *
617  * @param element JSON element
618  * @param expectedCount expected number of properties in element
619  */
620 inline void verifyPropertyCount(const nlohmann::json& element,
621                                 unsigned int expectedCount)
622 {
623     if (element.size() != expectedCount)
624     {
625         throw std::invalid_argument{"Element contains an invalid property"};
626     }
627 }
628 
629 } // namespace internal
630 
631 } // namespace phosphor::power::regulators::config_file_parser
632