jingle/check.c
author Nicolas Cornu <nicolas.cornu@ensi-bourges.fr>
Sun, 06 Jun 2010 23:19:47 +0200
changeset 14 77966ed56e14
child 16 cb085682970f
permissions -rw-r--r--
Create two subdirectories. One will contain the main jingle module and the other the file transfert app.

/*
 * check.c
 *
 * Copyrigth (C) 2010 Nicolas Cornu <nicolas.cornu@ensi-bourges.fr>
 *
 * This program 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 2 of the License, or (at
 * your option) any later version.
 *
 * This program 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 this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 * USA
 */

#include <glib.h>
#include <loudmouth/loudmouth.h>

#include "check.h"
#include "jingle.h"


JingleContentNode *check_content(LmMessageNode *node, GError **err);
gint index_in_array(const gchar *str, const gchar **array);



const gchar *jingle_content_creator[] = {
  "initiator",
  "responder",
  NULL
};

const gchar *jingle_content_senders[] = {
  "both",
  "initiator",
  "none",
  "responder",
  NULL
};


/**
 * Populate a jingle_data struct from a <jingle> element.
 * Check if the element is in compliance with the XEP.
 */
gboolean check_jingle(LmMessageNode *node, JingleNode *jn, GError **err)
{
  //gint nb_reason = 0;
  LmMessageNode *child = NULL;
  const gchar *actionstr;
  JingleContentNode *cn;

  actionstr     = lm_message_node_get_attribute(node, "action");
  jn->initiator = lm_message_node_get_attribute(node, "initiator");
  jn->responder = lm_message_node_get_attribute(node, "responder");
  jn->sid       = lm_message_node_get_attribute(node, "sid");

  if (actionstr == NULL || jn->sid == NULL) {
    g_set_error(err, JINGLE_CHECK_ERROR, JINGLE_CHECK_ERROR_MISSING,
                "an attribute of the jingle element is missing");
    return FALSE;
  }

  jn->action = jingle_action_from_str(actionstr);
  if (jn->action == JINGLE_UNKNOWN_ACTION) {
    g_set_error(err, JINGLE_CHECK_ERROR, JINGLE_CHECK_ERROR_BADVALUE,
                "the action attribute is invalid");
    return FALSE;
  }

  /*for (child = node->children; child; child = child->next) {
    if (!g_strcmp0(child->name, "reason"))
      nb_reason++;
  }

  if (nb_reason > 1) {
    g_set_error(err, JINGLE_CHECK_ERROR, JINGLE_CHECK_ERROR_BADELEM,
                "too many reason elements");
    return FALSE;
  }*/
  
  for (child = node->children; child; child = child->next) {
    if (!g_strcmp0(child->name, "content")) {
      cn = check_content(node, err);
      if(cn == NULL) {
        g_assert (*err != NULL);
        return FALSE;
	  }
	  jn->content = g_list_append(jn->content, cn);
    }
  }

  return TRUE;
}

JingleContentNode *check_content(LmMessageNode *node, GError **err)
{
  JingleContentNode *cn = g_new0(JingleContentNode, 1);
  const gchar *creatorstr, *sendersstr;
  gint tmp, tmp2;

  creatorstr      = lm_message_node_get_attribute(node, "action");
  cn->disposition = lm_message_node_get_attribute(node, "disposition");
  cn->name        = lm_message_node_get_attribute(node, "name");
  sendersstr      = lm_message_node_get_attribute(node, "senders");

  if (creatorstr == NULL || cn->name == NULL) {
    g_set_error(err, JINGLE_CHECK_ERROR, JINGLE_CHECK_ERROR_MISSING,
                "an attribute of the content element is missing");
    g_free(cn);
    return NULL;
  }
  
  tmp = index_in_array(creatorstr, jingle_content_creator);
  tmp2 = index_in_array(sendersstr, jingle_content_senders);
  if (tmp < 0 || tmp2 < 0) {
    g_set_error(err, JINGLE_CHECK_ERROR, JINGLE_CHECK_ERROR_BADVALUE,
                "the attribute creator or sender is invalid");
    g_free(cn);
    return NULL;
  }
  cn->creator = (JingleCreator)tmp;
  cn->senders = (JingleSenders)tmp2;
  

  cn->description = lm_message_node_get_child(node, "description");
  cn->transport   = lm_message_node_get_child(node, "transport");
  if (cn->description == NULL || cn->transport == NULL) {
    g_set_error(err, JINGLE_CHECK_ERROR, JINGLE_CHECK_ERROR_MISSING,
                "a child element of content is missing");
    g_free(cn);
    return NULL;
  }
  
  return cn;
}

gint index_in_array(const gchar *str, const gchar **array)
{
  gint i;
  for (i = 0; array[i]; i++) {
    if (!g_strcmp0(array[i], str)) {
      return i;
    }
  }
  return -1;
}

GQuark jingle_check_error_quark()
{
  return g_quark_from_string("JINGLE_CHECK_ERROR");
}