1
0
mirror of https://github.com/profanity-im/profanity.git synced 2024-12-04 14:46:46 -05:00
profanity/src/xmpp/form.c

611 lines
18 KiB
C
Raw Normal View History

2014-09-05 16:04:16 -04:00
/*
* form.c
*
* Copyright (C) 2012 - 2014 James Booth <boothj5@gmail.com>
*
* This file is part of Profanity.
*
* Profanity is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Profanity is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Profanity. If not, see <http://www.gnu.org/licenses/>.
*
* In addition, as a special exception, the copyright holders give permission to
* link the code of portions of this program with the OpenSSL library under
* certain conditions as described in each individual source file, and
* distribute linked combinations including the two.
*
* You must obey the GNU General Public License in all respects for all of the
* code used other than OpenSSL. If you modify file(s) with this exception, you
* may extend this exception to your version of the file(s), but you are not
* obligated to do so. If you do not wish to do so, delete this exception
* statement from your version. If you delete this exception statement from all
* source files in the program, then also delete it here.
*
*/
#include <string.h>
#include <stdlib.h>
#include <strophe.h>
2014-09-06 17:40:57 -04:00
#include <glib.h>
2014-09-05 16:04:16 -04:00
2014-09-06 17:40:57 -04:00
#include "log.h"
2014-09-05 16:04:16 -04:00
#include "xmpp/xmpp.h"
2014-09-06 17:40:57 -04:00
#include "xmpp/stanza.h"
2014-09-05 16:04:16 -04:00
#include "xmpp/connection.h"
2014-09-06 17:40:57 -04:00
static gboolean
_is_valid_form_element(xmpp_stanza_t *stanza)
{
char *name = xmpp_stanza_get_name(stanza);
if (g_strcmp0(name, STANZA_NAME_X) != 0) {
log_error("Error parsing form, root element not <x/>.");
return FALSE;
}
2014-09-05 16:04:16 -04:00
2014-09-06 17:40:57 -04:00
char *ns = xmpp_stanza_get_ns(stanza);
if (g_strcmp0(ns, STANZA_NS_DATA) != 0) {
log_error("Error parsing form, namespace not %s.", STANZA_NS_DATA);
return FALSE;
}
char *type = xmpp_stanza_get_attribute(stanza, STANZA_ATTR_TYPE);
if ((g_strcmp0(type, "form") != 0) &&
(g_strcmp0(type, "submit") != 0) &&
(g_strcmp0(type, "cancel") != 0) &&
(g_strcmp0(type, "result") != 0)) {
log_error("Error parsing form, unknown type.");
return FALSE;
}
return TRUE;
}
static DataForm *
_form_new(void)
2014-09-05 16:04:16 -04:00
{
2014-09-06 17:40:57 -04:00
DataForm *form = malloc(sizeof(DataForm));
form->type = NULL;
form->title = NULL;
form->instructions = NULL;
form->fields = NULL;
2014-09-11 18:44:39 -04:00
form->var_to_tag = NULL;
form->tag_to_var = NULL;
2014-09-13 16:36:16 -04:00
form->tag_ac = NULL;
2014-09-06 17:40:57 -04:00
return form;
}
static FormField *
_field_new(void)
{
FormField *field = malloc(sizeof(FormField));
field->label = NULL;
field->type = NULL;
field->var = NULL;
field->description = NULL;
field->required = FALSE;
field->values = NULL;
field->options = NULL;
return field;
}
static char *
_get_property(xmpp_stanza_t * const stanza, const char * const property)
{
char *result = NULL;
2014-09-05 16:04:16 -04:00
xmpp_ctx_t *ctx = connection_get_ctx();
2014-09-06 17:40:57 -04:00
xmpp_stanza_t *child = xmpp_stanza_get_child_by_name(stanza, property);
if (child != NULL) {
char *child_text = xmpp_stanza_get_text(child);
if (child_text != NULL) {
result = strdup(child_text);
xmpp_free(ctx, child_text);
}
}
return result;
}
static char *
_get_attr(xmpp_stanza_t * const stanza, const char * const attr)
{
char *result = xmpp_stanza_get_attribute(stanza, attr);
if (result != NULL) {
return strdup(result);
} else {
return NULL;
}
}
2014-09-05 16:04:16 -04:00
2014-09-06 17:40:57 -04:00
static gboolean
_is_required(xmpp_stanza_t * const stanza)
{
xmpp_stanza_t *child = xmpp_stanza_get_child_by_name(stanza, "required");
2014-09-05 16:04:16 -04:00
if (child != NULL) {
2014-09-06 17:40:57 -04:00
return TRUE;
} else {
return FALSE;
2014-09-05 16:04:16 -04:00
}
2014-09-06 17:40:57 -04:00
}
2014-09-05 16:04:16 -04:00
2014-09-10 08:18:36 -04:00
static form_field_type_t
_get_field_type(const char * const type)
{
if (g_strcmp0(type, "hidden") == 0) {
return FIELD_HIDDEN;
}
if (g_strcmp0(type, "text-single") == 0) {
return FIELD_TEXT_SINGLE;
}
if (g_strcmp0(type, "text-private") == 0) {
return FIELD_TEXT_PRIVATE;
}
if (g_strcmp0(type, "text-multi") == 0) {
return FIELD_TEXT_MULTI;
}
if (g_strcmp0(type, "boolean") == 0) {
return FIELD_BOOLEAN;
}
if (g_strcmp0(type, "list-single") == 0) {
return FIELD_LIST_SINGLE;
}
if (g_strcmp0(type, "list-multi") == 0) {
2014-09-13 19:18:06 -04:00
return FIELD_LIST_MULTI;
2014-09-10 08:18:36 -04:00
}
if (g_strcmp0(type, "jid-single") == 0) {
return FIELD_JID_SINGLE;
}
if (g_strcmp0(type, "jid-multi") == 0) {
return FIELD_JID_MULTI;
}
if (g_strcmp0(type, "fixed") == 0) {
return FIELD_FIXED;
}
return FIELD_UNKNOWN;
}
2014-09-06 17:40:57 -04:00
DataForm *
form_create(xmpp_stanza_t * const form_stanza)
{
xmpp_ctx_t *ctx = connection_get_ctx();
2014-09-05 16:04:16 -04:00
2014-09-06 17:40:57 -04:00
if (!_is_valid_form_element(form_stanza)) {
return NULL;
}
2014-09-05 16:04:16 -04:00
2014-09-06 17:40:57 -04:00
DataForm *form = _form_new();
form->type = _get_attr(form_stanza, "type");
form->title = _get_property(form_stanza, "title");
form->instructions = _get_property(form_stanza, "instructions");
2014-09-11 18:44:39 -04:00
form->var_to_tag = g_hash_table_new_full(g_str_hash, g_str_equal, free, free);
form->tag_to_var = g_hash_table_new_full(g_str_hash, g_str_equal, free, free);
form->tag_ac = autocomplete_new();
form->modified = FALSE;
2014-09-11 18:44:39 -04:00
int tag_num = 1;
2014-09-06 17:40:57 -04:00
// get fields
xmpp_stanza_t *form_child = xmpp_stanza_get_children(form_stanza);
while (form_child != NULL) {
char *child_name = xmpp_stanza_get_name(form_child);
if (g_strcmp0(child_name, "field") == 0) {
xmpp_stanza_t *field_stanza = form_child;
FormField *field = _field_new();
field->label = _get_attr(field_stanza, "label");
field->type = _get_attr(field_stanza, "type");
2014-09-10 08:18:36 -04:00
field->type_t = _get_field_type(field->type);
2014-09-11 18:44:39 -04:00
2014-09-06 17:40:57 -04:00
field->var = _get_attr(field_stanza, "var");
2014-09-13 16:46:54 -04:00
if (field->type_t != FIELD_HIDDEN) {
GString *tag = g_string_new("");
g_string_printf(tag, "field%d", tag_num++);
g_hash_table_insert(form->var_to_tag, strdup(field->var), strdup(tag->str));
g_hash_table_insert(form->tag_to_var, strdup(tag->str), strdup(field->var));
autocomplete_add(form->tag_ac, tag->str);
g_string_free(tag, TRUE);
}
2014-09-11 18:44:39 -04:00
2014-09-06 17:40:57 -04:00
field->description = _get_property(field_stanza, "desc");
field->required = _is_required(field_stanza);
// handle repeated field children
xmpp_stanza_t *field_child = xmpp_stanza_get_children(field_stanza);
while (field_child != NULL) {
child_name = xmpp_stanza_get_name(field_child);
2014-09-06 17:40:57 -04:00
// handle values
if (g_strcmp0(child_name, "value") == 0) {
char *value = xmpp_stanza_get_text(field_child);
if (value != NULL) {
field->values = g_slist_append(field->values, strdup(value));
2014-09-06 17:40:57 -04:00
xmpp_free(ctx, value);
}
// handle options
} else if (g_strcmp0(child_name, "option") == 0) {
FormOption *option = malloc(sizeof(FormOption));
option->label = _get_attr(field_child, "label");
option->value = _get_property(field_child, "value");
2014-09-05 16:04:16 -04:00
2014-09-06 17:40:57 -04:00
field->options = g_slist_append(field->options, option);
2014-09-05 16:04:16 -04:00
}
2014-09-06 17:40:57 -04:00
field_child = xmpp_stanza_get_next(field_child);
2014-09-05 16:04:16 -04:00
}
2014-09-06 17:40:57 -04:00
form->fields = g_slist_append(form->fields, field);
2014-09-05 16:04:16 -04:00
}
2014-09-06 17:40:57 -04:00
form_child = xmpp_stanza_get_next(form_child);
2014-09-05 16:04:16 -04:00
}
2014-09-06 17:40:57 -04:00
return form;
2014-09-05 16:04:16 -04:00
}
xmpp_stanza_t *
form_create_submission(DataForm *form)
{
xmpp_ctx_t *ctx = connection_get_ctx();
xmpp_stanza_t *x = xmpp_stanza_new(ctx);
xmpp_stanza_set_name(x, STANZA_NAME_X);
xmpp_stanza_set_ns(x, STANZA_NS_DATA);
xmpp_stanza_set_type(x, "submit");
GSList *curr_field = form->fields;
while (curr_field != NULL) {
FormField *field = curr_field->data;
xmpp_stanza_t *field_stanza = xmpp_stanza_new(ctx);
xmpp_stanza_set_name(field_stanza, "field");
xmpp_stanza_set_attribute(field_stanza, "var", field->var);
xmpp_stanza_t *value_stanza = NULL;
GSList *curr_value = NULL;
switch (field->type_t) {
2014-09-13 17:39:06 -04:00
case FIELD_HIDDEN:
case FIELD_FIXED:
case FIELD_TEXT_SINGLE:
case FIELD_TEXT_PRIVATE:
case FIELD_BOOLEAN:
case FIELD_LIST_SINGLE:
case FIELD_JID_SINGLE:
value_stanza = xmpp_stanza_new(ctx);
xmpp_stanza_set_name(value_stanza, "value");
if (field->values != NULL) {
if (field->values->data != NULL) {
xmpp_stanza_t *text_stanza = xmpp_stanza_new(ctx);
xmpp_stanza_set_text(text_stanza, field->values->data);
xmpp_stanza_add_child(value_stanza, text_stanza);
xmpp_stanza_release(text_stanza);
}
}
xmpp_stanza_add_child(field_stanza, value_stanza);
xmpp_stanza_release(value_stanza);
break;
case FIELD_TEXT_MULTI:
2014-09-13 19:18:06 -04:00
case FIELD_LIST_MULTI:
case FIELD_JID_MULTI:
curr_value = field->values;
while (curr_value != NULL) {
char *value = curr_value->data;
value_stanza = xmpp_stanza_new(ctx);
xmpp_stanza_set_name(value_stanza, "value");
if (value != NULL) {
xmpp_stanza_t *text_stanza = xmpp_stanza_new(ctx);
xmpp_stanza_set_text(text_stanza, value);
xmpp_stanza_add_child(value_stanza, text_stanza);
xmpp_stanza_release(text_stanza);
}
xmpp_stanza_add_child(field_stanza, value_stanza);
xmpp_stanza_release(value_stanza);
curr_value = g_slist_next(curr_value);
}
break;
default:
break;
}
xmpp_stanza_add_child(x, field_stanza);
xmpp_stanza_release(field_stanza);
curr_field = g_slist_next(curr_field);
}
return x;
}
2014-09-06 17:40:57 -04:00
static void
_free_option(FormOption *option)
{
if (option != NULL) {
free(option->label);
free(option->value);
2014-09-06 17:40:57 -04:00
free(option);
}
}
static void
_free_field(FormField *field)
{
if (field != NULL) {
free(field->label);
free(field->type);
free(field->var);
free(field->description);
g_slist_free_full(field->values, free);
g_slist_free_full(field->options, (GDestroyNotify)_free_option);
2014-09-06 17:40:57 -04:00
free(field);
}
}
static void
2014-09-05 16:04:16 -04:00
_form_destroy(DataForm *form)
{
if (form != NULL) {
free(form->type);
free(form->title);
free(form->instructions);
g_slist_free_full(form->fields, (GDestroyNotify)_free_field);
2014-09-11 18:44:39 -04:00
g_hash_table_destroy(form->var_to_tag);
g_hash_table_destroy(form->tag_to_var);
autocomplete_free(form->tag_ac);
2014-09-05 16:04:16 -04:00
free(form);
}
}
2014-09-06 17:40:57 -04:00
static char *
_form_get_form_type_field(DataForm *form)
2014-09-05 16:04:16 -04:00
{
2014-09-06 17:40:57 -04:00
GSList *curr = form->fields;
while (curr != NULL) {
FormField *field = curr->data;
if (g_strcmp0(field->var, "FORM_TYPE") == 0) {
2014-09-06 17:40:57 -04:00
return field->values->data;
}
curr = g_slist_next(curr);
}
return NULL;
2014-09-05 16:04:16 -04:00
}
static gboolean
_form_tag_exists(DataForm *form, const char * const tag)
{
GList *tags = g_hash_table_get_keys(form->tag_to_var);
GList *curr = tags;
while (curr != NULL) {
if (g_strcmp0(curr->data, tag) == 0) {
return TRUE;
}
curr = g_list_next(curr);
}
return FALSE;
}
2014-09-13 17:39:06 -04:00
static form_field_type_t
2014-09-13 19:18:06 -04:00
_form_get_field_type(DataForm *form, const char * const tag)
2014-09-13 17:39:06 -04:00
{
char *var = g_hash_table_lookup(form->tag_to_var, tag);
if (var != NULL) {
GSList *curr = form->fields;
while (curr != NULL) {
FormField *field = curr->data;
if (g_strcmp0(field->var, var) == 0) {
return field->type_t;
}
curr = g_slist_next(curr);
}
}
return FIELD_UNKNOWN;
}
static void
2014-09-13 19:18:06 -04:00
_form_set_value(DataForm *form, const char * const tag, char *value)
{
char *var = g_hash_table_lookup(form->tag_to_var, tag);
if (var != NULL) {
GSList *curr = form->fields;
while (curr != NULL) {
FormField *field = curr->data;
if (g_strcmp0(field->var, var) == 0) {
if (g_slist_length(field->values) == 0) {
field->values = g_slist_append(field->values, strdup(value));
form->modified = TRUE;
return;
} else if (g_slist_length(field->values) == 1) {
free(field->values->data);
field->values->data = strdup(value);
form->modified = TRUE;
return;
}
}
curr = g_slist_next(curr);
}
}
}
2014-09-13 19:18:06 -04:00
static void
_form_add_value(DataForm *form, const char * const tag, char *value)
{
char *var = g_hash_table_lookup(form->tag_to_var, tag);
if (var != NULL) {
GSList *curr = form->fields;
while (curr != NULL) {
FormField *field = curr->data;
if (g_strcmp0(field->var, var) == 0) {
field->values = g_slist_append(field->values, strdup(value));
form->modified = TRUE;
return;
}
curr = g_slist_next(curr);
}
}
}
static gboolean
_form_add_unique_value(DataForm *form, const char * const tag, char *value)
{
char *var = g_hash_table_lookup(form->tag_to_var, tag);
if (var != NULL) {
GSList *curr = form->fields;
while (curr != NULL) {
FormField *field = curr->data;
if (g_strcmp0(field->var, var) == 0) {
2014-09-13 19:18:06 -04:00
GSList *curr_value = field->values;
while (curr_value != NULL) {
if (g_strcmp0(curr_value->data, value) == 0) {
return FALSE;
2014-09-13 19:18:06 -04:00
}
curr_value = g_slist_next(curr_value);
}
field->values = g_slist_append(field->values, strdup(value));
form->modified = TRUE;
return TRUE;
2014-09-13 19:18:06 -04:00
}
curr = g_slist_next(curr);
}
}
return FALSE;
2014-09-13 19:18:06 -04:00
}
static gboolean
_form_remove_value(DataForm *form, const char * const tag, char *value)
{
char *var = g_hash_table_lookup(form->tag_to_var, tag);
if (var != NULL) {
GSList *curr = form->fields;
while (curr != NULL) {
FormField *field = curr->data;
if (g_strcmp0(field->var, var) == 0) {
GSList *found = g_slist_find_custom(field->values, value, (GCompareFunc)g_strcmp0);
if (found != NULL) {
free(found->data);
found->data = NULL;
field->values = g_slist_delete_link(field->values, found);
form->modified = TRUE;
return TRUE;
} else {
return FALSE;
}
}
curr = g_slist_next(curr);
}
}
return FALSE;
}
static gboolean
_form_remove_text_multi_value(DataForm *form, const char * const tag, int index)
{
index--;
char *var = g_hash_table_lookup(form->tag_to_var, tag);
if (var != NULL) {
GSList *curr = form->fields;
while (curr != NULL) {
FormField *field = curr->data;
if (g_strcmp0(field->var, var) == 0) {
GSList *item = g_slist_nth(field->values, index);
if (item != NULL) {
free(item->data);
item->data = NULL;
field->values = g_slist_delete_link(field->values, item);
form->modified = TRUE;
return TRUE;
} else {
return FALSE;
}
}
curr = g_slist_next(curr);
}
}
return FALSE;
}
static int
_form_get_value_count(DataForm *form, const char * const tag)
{
char *var = g_hash_table_lookup(form->tag_to_var, tag);
if (var != NULL) {
GSList *curr = form->fields;
while (curr != NULL) {
FormField *field = curr->data;
if (g_strcmp0(field->var, var) == 0) {
if ((g_slist_length(field->values) == 1) && (field->values->data == NULL)) {
return 0;
} else {
return g_slist_length(field->values);
}
}
curr = g_slist_next(curr);
}
}
return 0;
}
static gboolean
2014-09-13 19:18:06 -04:00
_form_field_contains_option(DataForm *form, const char * const tag, char *value)
{
char *var = g_hash_table_lookup(form->tag_to_var, tag);
if (var != NULL) {
GSList *curr = form->fields;
while (curr != NULL) {
FormField *field = curr->data;
if (g_strcmp0(field->var, var) == 0) {
GSList *curr_option = field->options;
while (curr_option != NULL) {
FormOption *option = curr_option->data;
if (g_strcmp0(option->value, value) == 0) {
return TRUE;
}
curr_option = g_slist_next(curr_option);
}
}
curr = g_slist_next(curr);
}
}
return FALSE;
}
2014-09-05 16:04:16 -04:00
void
form_init_module(void)
{
form_destroy = _form_destroy;
form_get_form_type_field = _form_get_form_type_field;
2014-09-13 19:18:06 -04:00
form_get_field_type = _form_get_field_type;
form_set_value = _form_set_value;
form_add_unique_value = _form_add_unique_value;
2014-09-13 19:18:06 -04:00
form_add_value = _form_add_value;
form_remove_value = _form_remove_value;
form_remove_text_multi_value = _form_remove_text_multi_value;
2014-09-13 19:18:06 -04:00
form_field_contains_option = _form_field_contains_option;
form_tag_exists = _form_tag_exists;
form_get_value_count = _form_get_value_count;
2014-09-05 16:04:16 -04:00
}