aboutsummaryrefslogtreecommitdiffstats
path: root/src/libserver
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2017-04-22 14:25:24 +0100
committerVsevolod Stakhov <vsevolod@highsecure.ru>2017-04-22 14:56:37 +0100
commit9c2a259a7c4c1251b51b4cfc54e49f639d80e0c4 (patch)
treee27d971bcc11e9c1e560d387dada32e82b53b907 /src/libserver
parentcae47edfb5950a877ea94aefd716fbcbb1e5c3a1 (diff)
downloadrspamd-9c2a259a7c4c1251b51b4cfc54e49f639d80e0c4.tar.gz
rspamd-9c2a259a7c4c1251b51b4cfc54e49f639d80e0c4.zip
[Feature] Add ability to add doc strings by example
Diffstat (limited to 'src/libserver')
-rw-r--r--src/libserver/cfg_rcl.c71
-rw-r--r--src/libserver/cfg_rcl.h27
2 files changed, 98 insertions, 0 deletions
diff --git a/src/libserver/cfg_rcl.c b/src/libserver/cfg_rcl.c
index 70cb36240..b6adc61a8 100644
--- a/src/libserver/cfg_rcl.c
+++ b/src/libserver/cfg_rcl.c
@@ -3863,3 +3863,74 @@ rspamd_rcl_add_doc_by_path (struct rspamd_config *cfg,
default_value,
required);
}
+
+static void
+rspamd_rcl_add_doc_from_comments (struct rspamd_config *cfg,
+ ucl_object_t *top_doc, const ucl_object_t *obj,
+ const ucl_object_t *comments, gboolean is_top)
+{
+ ucl_object_iter_t it;
+ const ucl_object_t *cur, *cmt;
+ ucl_object_t *cur_doc;
+
+ if (ucl_object_type (obj) == UCL_OBJECT) {
+ while ((cur = ucl_object_iterate (obj, &it, true)) != NULL) {
+ cur_doc = NULL;
+
+ if ((cmt = ucl_comments_find (comments, cur)) != NULL) {
+ cur_doc = rspamd_rcl_add_doc_obj (top_doc,
+ ucl_object_tostring (cmt), ucl_object_key (cur),
+ ucl_object_type (cur), NULL, 0, NULL, FALSE);
+ }
+
+ if (cur_doc && ucl_object_type (cur) == UCL_OBJECT) {
+ rspamd_rcl_add_doc_from_comments (cfg, cur_doc, cur, comments,
+ FALSE);
+ }
+ }
+ }
+ else if (!is_top) {
+ if ((cmt = ucl_comments_find (comments, obj)) != NULL) {
+ rspamd_rcl_add_doc_obj (top_doc,
+ ucl_object_tostring (cmt), ucl_object_key (obj),
+ ucl_object_type (obj), NULL, 0, NULL, FALSE);
+ }
+
+ }
+}
+
+ucl_object_t *
+rspamd_rcl_add_doc_by_example (struct rspamd_config *cfg,
+ const gchar *root_path,
+ const gchar *doc_string,
+ const gchar *doc_name,
+ const gchar *example_data, gsize example_len)
+{
+ struct ucl_parser *parser;
+ ucl_object_t *top, *top_doc;
+ const ucl_object_t *comments;
+
+ parser = ucl_parser_new (UCL_PARSER_NO_FILEVARS|UCL_PARSER_SAVE_COMMENTS);
+
+ if (!ucl_parser_add_chunk (parser, example_data, example_len)) {
+ msg_err_config ("cannot parse example: %s",
+ ucl_parser_get_error (parser));
+ ucl_parser_free (parser);
+
+ return NULL;
+ }
+
+ top = ucl_parser_get_object (parser);
+ comments = ucl_parser_get_comments (parser);
+
+ /* Add top object */
+ top_doc = rspamd_rcl_add_doc_by_path (cfg, root_path, doc_string,
+ doc_name, ucl_object_type (top), NULL, 0, NULL, FALSE);
+ ucl_object_insert_key (top_doc,
+ ucl_object_fromlstring (example_data, example_len),
+ "example", 0, false);
+
+ rspamd_rcl_add_doc_from_comments (cfg, top_doc, top, comments, TRUE);
+
+ return top_doc;
+}
diff --git a/src/libserver/cfg_rcl.h b/src/libserver/cfg_rcl.h
index 6a0d86cb7..a83e9698f 100644
--- a/src/libserver/cfg_rcl.h
+++ b/src/libserver/cfg_rcl.h
@@ -419,4 +419,31 @@ ucl_object_t *rspamd_rcl_add_doc_by_path (struct rspamd_config *cfg,
gint flags,
const char *default_value,
gboolean required);
+
+/**
+ * Parses example and adds documentation according to the example:
+ *
+ * ```
+ * section {
+ * param1 = value; # explanation
+ * param2 = value; # explanation
+ * }
+ * ```
+ *
+ * will produce the following documentation strings:
+ * section ->
+ * section.param1 : explanation
+ * section.param2 : explanation
+ *
+ * @param cfg
+ * @param root_path
+ * @param example_data
+ * @param example_len
+ * @return
+ */
+ucl_object_t *rspamd_rcl_add_doc_by_example (struct rspamd_config *cfg,
+ const gchar *root_path,
+ const gchar *doc_string,
+ const gchar *doc_name,
+ const gchar *example_data, gsize example_len);
#endif /* CFG_RCL_H_ */