Linux Audio

Check our new training course

Embedded Linux Audio

Check our new training course
with Creative Commons CC-BY-SA
lecture materials

Bootlin logo

Elixir Cross Referencer

Loading...
/*
 * Copyright (c) 2013, Institute for Pervasive Computing, ETH Zurich
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 * 3. Neither the name of the Institute nor the names of its contributors
 *    may be used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 *
 * This file is part of the Contiki operating system.
 */

/**
 * \file
 *      An abstraction layer for RESTful Web services (Erbium).
 *      Inspired by RESTful Contiki by Dogan Yazar.
 * \author
 *      Matthias Kovatsch <kovatsch@inf.ethz.ch>
 */

#ifndef REST_ENGINE_H_
#define REST_ENGINE_H_

#include <stdio.h>
#include "contiki.h"
#include "contiki-lib.h"
#include "rest-constants.h"

/* list of valid REST Enigne implementations */
#define REGISTERED_ENGINE_ERBIUM coap_rest_implementation
#define REGISTERED_ENGINE_HELIUM http_rest_implementation

/* sanity check for configured implementation */
#if !defined(REST) || (REST != REGISTERED_ENGINE_ERBIUM && REST != REGISTERED_ENGINE_HELIUM)
#error "Define a valid REST Engine implementation (REST define)!"
#endif

/*
 * The maximum buffer size that is provided for resource responses and must be respected due to the limited IP buffer.
 * Larger data must be handled by the resource and will be sent chunk-wise through a TCP stream or CoAP blocks.
 */
#ifndef REST_MAX_CHUNK_SIZE
#define REST_MAX_CHUNK_SIZE     64
#endif

#ifndef MIN
#define MIN(a, b) ((a) < (b) ? (a) : (b))
#endif /* MIN */

struct resource_s;
struct periodic_resource_s;

/* signatures of handler functions */
typedef void (*restful_handler)(void *request, void *response,
                                uint8_t *buffer, uint16_t preferred_size,
                                int32_t *offset);
typedef void (*restful_final_handler)(struct resource_s *resource,
                                      void *request, void *response);
typedef void (*restful_periodic_handler)(void);
typedef void (*restful_response_handler)(void *data, void *response);
typedef void (*restful_trigger_handler)(void);

/* signature of the rest-engine service function */
typedef int (*service_callback_t)(void *request, void *response,
                                  uint8_t *buffer, uint16_t preferred_size,
                                  int32_t *offset);

/* data structure representing a resource in REST */
struct resource_s {
  struct resource_s *next;        /* for LIST, points to next resource defined */
  const char *url;                /*handled URL */
  rest_resource_flags_t flags;    /* handled RESTful methods */
  const char *attributes;         /* link-format attributes */
  restful_handler get_handler;    /* handler function */
  restful_handler post_handler;   /* handler function */
  restful_handler put_handler;    /* handler function */
  restful_handler delete_handler; /* handler function */
  union {
    struct periodic_resource_s *periodic; /* special data depending on flags */
    restful_trigger_handler trigger;
    restful_trigger_handler resume;
  };
};
typedef struct resource_s resource_t;

struct periodic_resource_s {
  struct periodic_resource_s *next; /* for LIST, points to next resource defined */
  const resource_t *resource;
  uint32_t period;
  struct etimer periodic_timer;
  const restful_periodic_handler periodic_handler;
};
typedef struct periodic_resource_s periodic_resource_t;

/*
 * Macro to define a RESTful resource.
 * Resources are statically defined for the sake of efficiency and better memory management.
 */
#define RESOURCE(name, attributes, get_handler, post_handler, put_handler, delete_handler) \
  resource_t name = { NULL, NULL, NO_FLAGS, attributes, get_handler, post_handler, put_handler, delete_handler, { NULL } }

#define PARENT_RESOURCE(name, attributes, get_handler, post_handler, put_handler, delete_handler) \
  resource_t name = { NULL, NULL, HAS_SUB_RESOURCES, attributes, get_handler, post_handler, put_handler, delete_handler, { NULL } }

#define SEPARATE_RESOURCE(name, attributes, get_handler, post_handler, put_handler, delete_handler, resume_handler) \
  resource_t name = { NULL, NULL, IS_SEPARATE, attributes, get_handler, post_handler, put_handler, delete_handler, { .resume = resume_handler } }

#define EVENT_RESOURCE(name, attributes, get_handler, post_handler, put_handler, delete_handler, event_handler) \
  resource_t name = { NULL, NULL, IS_OBSERVABLE, attributes, get_handler, post_handler, put_handler, delete_handler, { .trigger = event_handler } }

/*
 * Macro to define a periodic resource.
 * The corresponding [name]_periodic_handler() function will be called every period.
 * For instance polling a sensor and publishing a changed value to subscribed clients would be done there.
 * The subscriber list will be maintained by the final_handler rest_subscription_handler() (see rest-mapping header file).
 */
#define PERIODIC_RESOURCE(name, attributes, get_handler, post_handler, put_handler, delete_handler, period, periodic_handler) \
  periodic_resource_t periodic_##name; \
  resource_t name = { NULL, NULL, IS_OBSERVABLE | IS_PERIODIC, attributes, get_handler, post_handler, put_handler, delete_handler, { .periodic = &periodic_##name } }; \
  periodic_resource_t periodic_##name = { NULL, &name, period, { { 0 } }, periodic_handler };

struct rest_implementation {
  char *name;

  /** Initialize the REST implementation. */
  void (*init)(void);

  /** Register the RESTful service callback at implementation. */
  void (*set_service_callback)(service_callback_t callback);

  /** Get request URI path. */
  int (*get_url)(void *request, const char **url);

  /** Get the method of a request. */
  rest_resource_flags_t (*get_method_type)(void *request);

  /** Set the status code of a response. */
  int (*set_response_status)(void *response, unsigned int code);

  /** Get the content-type of a request. */
  int (*get_header_content_type)(void *request,
                                 unsigned int *content_format);

  /** Set the Content-Type of a response. */
  int (*set_header_content_type)(void *response,
                                 unsigned int content_format);

  /** Get the Accept types of a request. */
  int (*get_header_accept)(void *request, unsigned int *accept);

  /** Get the Length option of a request. */
  int (*get_header_length)(void *request, uint32_t *size);

  /** Set the Length option of a response. */
  int (*set_header_length)(void *response, uint32_t size);

  /** Get the Max-Age option of a request. */
  int (*get_header_max_age)(void *request, uint32_t *age);

  /** Set the Max-Age option of a response. */
  int (*set_header_max_age)(void *response, uint32_t age);

  /** Set the ETag option of a response. */
  int (*set_header_etag)(void *response, const uint8_t *etag,
                         size_t length);

  /** Get the If-Match option of a request. */
  int (*get_header_if_match)(void *request, const uint8_t **etag);

  /** Get the If-Match option of a request. */
  int (*get_header_if_none_match)(void *request);

  /** Get the Host option of a request. */
  int (*get_header_host)(void *request, const char **host);

  /** Set the location option of a response. */
  int (*set_header_location)(void *response, const char *location);

  /** Get the payload option of a request. */
  int (*get_request_payload)(void *request, const uint8_t **payload);

  /** Set the payload option of a response. */
  int (*set_response_payload)(void *response, const void *payload,
                              size_t length);

  /** Get the query string of a request. */
  int (*get_query)(void *request, const char **value);

  /** Get the value of a request query key-value pair. */
  int (*get_query_variable)(void *request, const char *name,
                            const char **value);

  /** Get the value of a request POST key-value pair. */
  int (*get_post_variable)(void *request, const char *name,
                           const char **value);

  /** Send the payload to all subscribers of the resource at url. */
  void (*notify_subscribers)(resource_t *resource);

  /** The handler for resource subscriptions. */
  restful_final_handler subscription_handler;

  /* REST status codes. */
  const struct rest_implementation_status status;

  /* REST content-types. */
  const struct rest_implementation_type type;
};

/* instance of REST implementation */
extern const struct rest_implementation REST;

/*
 * To be called by HTTP/COAP server as a callback function when a new service request appears.
 * This function dispatches the corresponding RESTful service.
 */
int rest_invoke_restful_service(void *request, void *response,
                                uint8_t *buffer, uint16_t buffer_size,
                                int32_t *offset);
/*---------------------------------------------------------------------------*/
/**
 * \brief      Initializes REST framework and starts the HTTP or CoAP process.
 */
void rest_init_engine(void);
/*---------------------------------------------------------------------------*/
/**
 *
 * \brief      Resources wanted to be accessible should be activated with the following code.
 * \param resource
 *             A RESTful resource defined through the RESOURCE macros.
 * \param path
 *             The local URI path where to provide the resource.
 */
void rest_activate_resource(resource_t *resource, char *path);
/*---------------------------------------------------------------------------*/
/**
 * \brief      Returns the list of registered RESTful resources.
 * \return     The resource list.
 */
list_t rest_get_resources(void);
/*---------------------------------------------------------------------------*/

#endif /*REST_ENGINE_H_ */