libamxd  6.4.1
Data Model Manager
amxd_action_object_write.c
Go to the documentation of this file.
1 /****************************************************************************
2 **
3 ** SPDX-License-Identifier: BSD-2-Clause-Patent
4 **
5 ** SPDX-FileCopyrightText: Copyright (c) 2023 SoftAtHome
6 **
7 ** Redistribution and use in source and binary forms, with or without modification,
8 ** are permitted provided that the following conditions are met:
9 **
10 ** 1. Redistributions of source code must retain the above copyright notice,
11 ** this list of conditions and the following disclaimer.
12 **
13 ** 2. Redistributions in binary form must reproduce the above copyright notice,
14 ** this list of conditions and the following disclaimer in the documentation
15 ** and/or other materials provided with the distribution.
16 **
17 ** Subject to the terms and conditions of this license, each copyright holder
18 ** and contributor hereby grants to those receiving rights under this license
19 ** a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable
20 ** (except for failure to satisfy the conditions of this license) patent license
21 ** to make, have made, use, offer to sell, sell, import, and otherwise transfer
22 ** this software, where such license applies only to those patent claims, already
23 ** acquired or hereafter acquired, licensable by such copyright holder or contributor
24 ** that are necessarily infringed by:
25 **
26 ** (a) their Contribution(s) (the licensed copyrights of copyright holders and
27 ** non-copyrightable additions of contributors, in source or binary form) alone;
28 ** or
29 **
30 ** (b) combination of their Contribution(s) with the work of authorship to which
31 ** such Contribution(s) was added by such copyright holder or contributor, if,
32 ** at the time the Contribution is added, such addition causes such combination
33 ** to be necessarily infringed. The patent license shall not apply to any other
34 ** combinations which include the Contribution.
35 **
36 ** Except as expressly stated above, no rights or licenses from any copyright
37 ** holder or contributor is granted under this license, whether expressly, by
38 ** implication, estoppel or otherwise.
39 **
40 ** DISCLAIMER
41 **
42 ** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
43 ** AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
44 ** IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
45 ** ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE
46 ** LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
47 ** DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
48 ** SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
49 ** CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
50 ** OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
51 ** USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
52 **
53 ****************************************************************************/
54 
55 #include <stdlib.h>
56 
57 #include "amxd_priv.h"
58 
59 #include <amxd/amxd_dm.h>
60 #include <amxd/amxd_action.h>
61 #include <amxd/amxd_object.h>
62 #include <amxd/amxd_path.h>
63 
64 #include "amxd_assert.h"
65 
67  const amxd_param_t* const param,
68  bool set_read_only,
69  amxd_dm_access_t access) {
71 
72  when_true_status(amxd_param_is_attr_set(param, amxd_pattr_private) &&
73  access != amxd_dm_access_private,
74  exit,
76  when_true_status(object->type == amxd_object_instance &&
78  exit,
80  when_true_status(object->type == amxd_object_template &&
82  exit,
84  when_true_status(amxd_param_is_attr_set(param, amxd_pattr_read_only) &&
85  !set_read_only,
86  exit,
88 
89 exit:
90  return status;
91 }
92 
93 static amxc_var_t* amxd_set_action_get_result(const amxd_object_t* object, amxc_var_t* retval) {
95  amxc_var_t* result = GET_ARG(retval, path);
96 
97  if(result == NULL) {
98  result = amxc_var_add_new_key(retval, path);
99  amxc_var_set_type(result, AMXC_VAR_ID_HTABLE);
100  result = amxc_var_add_key(amxc_htable_t, result, "parameters", NULL);
101  } else {
102  result = GET_ARG(result, "parameters");
103  }
104 
105  free(path);
106  return result;
107 }
108 
109 static void amxd_set_action_add_error(amxc_var_t* result,
110  const char* name,
111  bool required,
113  result = amxc_var_add_key(amxc_htable_t, result, name, NULL);
114  amxc_var_add_key(uint32_t, result, "error_code", status);
115  amxc_var_add_key(bool, result, "required", required);
116 }
117 
118 static void amxd_set_action_add_result(amxc_var_t* result,
119  const char* name,
120  uint32_t type,
121  amxc_var_t* value) {
122  result = amxc_var_add_new_key(result, name);
123  amxc_var_copy(result, value);
124  amxc_var_cast(result, type);
125 }
126 
128  amxd_dm_access_t access,
129  const char* name,
130  amxc_var_t* value,
131  bool ro,
132  bool required,
133  amxc_var_t* ret) {
134  amxc_var_t* obj_result = NULL;
135  amxd_param_t* param = NULL;
137 
138  param = amxd_object_get_param_def(object, name);
139  if(param == NULL) {
140  obj_result = amxd_set_action_get_result(object, ret);
141  amxd_set_action_add_error(obj_result, name, required, amxd_status_parameter_not_found);
143  goto exit;
144  }
145  object = amxd_param_get_owner(param);
146  obj_result = amxd_set_action_get_result(object, ret);
147 
148  status = amxd_can_set_parameter(object, param, ro, access);
149  if(status != amxd_status_ok) {
150  amxd_set_action_add_error(obj_result, name, required, status);
151  goto exit;
152  }
153 
154  status = amxd_param_set_value(param, value);
155  if(status != amxd_status_ok) {
156  amxd_set_action_add_error(obj_result, name, required, status);
157  goto exit;
158  }
159 
160  amxd_set_action_add_result(obj_result, amxd_param_get_name(param), amxd_param_get_type(param), value);
161 
162 exit:
163  return status;
164 }
165 
166 bool PRIVATE amxd_param_is_unique(amxd_object_t* const templ,
167  amxd_object_t* const current,
168  amxd_param_t* const param,
169  const amxc_var_t* const value) {
170  bool unique = true;
173  const char* pname = amxd_param_get_name(param);
174  char* v = amxc_var_dyncast(cstring_t, value);
175  amxd_object_t* instance = amxd_object_findf(templ, "[%s == '%s']", pname, v);
176  free(v);
177  if((instance != NULL) && (instance != current)) {
178  unique = false;
179  }
180 
181  }
182 
183  return unique;
184 }
185 
187  amxd_dm_access_t access,
188  bool ro,
189  const amxc_var_t* values,
190  amxc_var_t* ret,
191  bool required) {
193 
194  amxc_var_for_each(value, values) {
195  const char* name = amxc_var_key(value);
197  if(amxc_var_type_of(value) == AMXC_VAR_ID_HTABLE) {
198  amxd_object_t* instance = NULL;
199  when_true_status(amxd_object_get_type(object) != amxd_object_template,
200  exit,
202  instance = amxd_object_get_instance(object, name, 0);
203  when_null_status(instance, exit, status = amxd_status_object_not_found);
204  ps = amxd_action_set_values(instance, access, ro, value, ret, required);
205  if(ps != amxd_status_ok) {
206  status = ps;
207  }
208  } else {
209  ps = amxd_action_set_value(object, access, name, value, ro, required, ret);
210  if(ps != amxd_status_ok) {
211  status = ps;
212  }
213  }
214  }
215 
216 exit:
217  return status;
218 }
219 
221  UNUSED amxd_param_t* const p,
222  amxd_action_t reason,
223  const amxc_var_t* const args,
224  amxc_var_t* const retval,
225  UNUSED void* priv) {
227  amxc_var_t* params = NULL;
229  bool set_read_only = false;
230 
231  when_null(object, exit);
232  when_null_status(args, exit, status = amxd_status_invalid_function_argument);
233  when_true_status(reason != action_object_write,
234  exit,
236 
237  when_true_status(amxc_var_type_of(args) != AMXC_VAR_ID_HTABLE,
238  exit,
240 
241  amxc_var_set_type(retval, AMXC_VAR_ID_HTABLE);
242  access = (amxd_dm_access_t) amxc_var_dyncast(uint32_t, GET_ARG(args, "access"));
243  set_read_only = GET_BOOL(args, "set_read_only");
244 
245  if(amxd_action_verify_access(object, access)) {
246  params = amxc_var_get_key(args, "parameters", AMXC_VAR_FLAG_DEFAULT);
247  if(params != NULL) {
248  when_true_status(amxc_var_type_of(params) != AMXC_VAR_ID_HTABLE,
249  exit,
251  status = amxd_action_set_values(object, access, set_read_only, params, retval, true);
252  when_failed(status, exit);
253  }
254  params = amxc_var_get_key(args, "oparameters", AMXC_VAR_FLAG_DEFAULT);
255  if(params != NULL) {
256  when_true_status(amxc_var_type_of(params) != AMXC_VAR_ID_HTABLE,
257  exit,
259  amxd_action_set_values(object, access, set_read_only, params, retval, false);
260  }
262  } else {
264  }
265 
266 exit:
267  return status;
268 }
269 
271  amxc_var_t* const values) {
272 
274  amxc_var_t args;
275 
276  amxc_var_init(&args);
277  when_null(object, exit);
278  when_true(amxc_var_is_null(values) ||
279  amxc_var_type_of(values) != AMXC_VAR_ID_HTABLE,
280  exit);
281 
282  amxc_var_set_type(&args, AMXC_VAR_ID_HTABLE);
283  amxc_var_set_key(&args, "parameters", values, AMXC_VAR_FLAG_COPY);
284  amxc_var_add_key(uint32_t, &args, "access", amxd_dm_access_private);
285  amxc_var_add_key(bool, &args, "set_read_only", true);
286  status = amxd_dm_invoke_action(object,
287  NULL,
289  &args,
290  NULL);
291 
292 exit:
293  amxc_var_clean(&args);
294  return status;
295 }
Ambiorix Data Model Default actions header file.
static amxd_status_t amxd_can_set_parameter(const amxd_object_t *const object, const amxd_param_t *const param, bool set_read_only, amxd_dm_access_t access)
amxd_status_t PRIVATE amxd_action_set_values(amxd_object_t *const object, amxd_dm_access_t access, bool ro, const amxc_var_t *values, amxc_var_t *ret, bool required)
amxd_status_t amxd_action_object_write(amxd_object_t *const object, UNUSED amxd_param_t *const p, amxd_action_t reason, const amxc_var_t *const args, amxc_var_t *const retval, UNUSED void *priv)
static amxd_status_t amxd_action_set_value(amxd_object_t *object, amxd_dm_access_t access, const char *name, amxc_var_t *value, bool ro, bool required, amxc_var_t *ret)
static void amxd_set_action_add_error(amxc_var_t *result, const char *name, bool required, amxd_status_t status)
static amxc_var_t * amxd_set_action_get_result(const amxd_object_t *object, amxc_var_t *retval)
bool PRIVATE amxd_param_is_unique(amxd_object_t *const templ, amxd_object_t *const current, amxd_param_t *const param, const amxc_var_t *const value)
static void amxd_set_action_add_result(amxc_var_t *result, const char *name, uint32_t type, amxc_var_t *value)
Ambiorix Data Model API header file.
amxd_status_t amxd_dm_invoke_action(amxd_object_t *object, amxd_param_t *param, amxd_action_t reason, const amxc_var_t *const args, amxc_var_t *const retval)
Definition: amxd_dm.c:591
Ambiorix Data Model API header file.
bool amxd_action_verify_access(amxd_object_t *object, amxd_dm_access_t access)
bool amxd_param_is_attr_set(const amxd_param_t *const param, const amxd_pattr_id_t attr)
const char * amxd_param_get_name(const amxd_param_t *const param)
amxd_status_t amxd_param_set_value(amxd_param_t *const param, const amxc_var_t *const value)
static uint32_t amxd_param_get_type(const amxd_param_t *const param)
amxd_object_t * amxd_param_get_owner(const amxd_param_t *const param)
Ambiorix path API header file.
@ amxd_pattr_template
Definition: amxd_types.h:355
@ amxd_pattr_private
Definition: amxd_types.h:357
@ amxd_pattr_instance
Definition: amxd_types.h:356
@ amxd_pattr_unique
Definition: amxd_types.h:363
@ amxd_pattr_key
Definition: amxd_types.h:362
@ amxd_pattr_read_only
Definition: amxd_types.h:358
@ action_object_write
Definition: amxd_types.h:118
enum _amxd_action amxd_action_t
enum _amxd_status amxd_status_t
@ amxd_status_read_only
Definition: amxd_types.h:93
@ amxd_status_parameter_not_found
Definition: amxd_types.h:82
@ amxd_status_function_not_implemented
Definition: amxd_types.h:83
@ amxd_status_object_not_found
Definition: amxd_types.h:80
@ amxd_status_invalid_function_argument
Definition: amxd_types.h:85
@ amxd_status_ok
Definition: amxd_types.h:78
@ amxd_status_unknown_error
Definition: amxd_types.h:79
#define AMXD_OBJECT_INDEXED
Name and path format flag - use index for instance objects.
Definition: amxd_object.h:176
enum _amxd_dm_access amxd_dm_access_t
Access level.
#define AMXD_OBJECT_TERMINATE
Path format flag - when set the object path is terminated with a dot.
Definition: amxd_object.h:214
@ amxd_dm_access_private
Definition: amxd_types.h:141
@ amxd_dm_access_public
Definition: amxd_types.h:136
@ amxd_object_template
Definition: amxd_types.h:183
@ amxd_object_instance
Definition: amxd_types.h:186
amxd_object_t amxd_status_t amxd_status_t char * amxd_object_get_path(const amxd_object_t *object, const uint32_t flags)
Get the full path of the object.
amxd_object_t * amxd_object_findf(amxd_object_t *object, const char *rel_path,...) __attribute__((format(printf
Find an object in the data model tree, starting from an object.
amxd_object_t * amxd_object_get_instance(const amxd_object_t *object, const char *name, uint32_t index)
Get an instance of the template object.
amxd_status_t amxd_object_set_params(amxd_object_t *const object, amxc_var_t *const values)
Sets multiple parameter values in a data model object.
amxd_param_t * amxd_object_get_param_def(const amxd_object_t *const object, const char *name)
Gets a parameter definition from an object.
static amxd_object_type_t amxd_object_get_type(const amxd_object_t *const object)
Returns the object type.
Definition: amxd_object.h:586
amxd_object_type_t type
Definition: amxd_types.h:236
static amxd_status_t status