From 015b82d9a8987f613c15671e480fa7b3c92bd5f5 Mon Sep 17 00:00:00 2001
From: "Dmitriy A. Gerasimov" <dmitriy.gerasimov@demlabs.net>
Date: Tue, 16 Feb 2021 00:59:33 +0700
Subject: [PATCH] [+] In-memory cache for dap_http objects [*] Some refactoring
 here, mostly renames and tabs

---
 dap-sdk/net/server/http_server/dap_http.c     | 133 +++++----
 .../net/server/http_server/dap_http_cache.c   |  69 +++++
 .../net/server/http_server/dap_http_simple.c  |  13 +
 .../http_server/http_client/dap_http_client.c | 267 ++++++++++--------
 .../http_server/http_client/dap_http_header.c |  41 ++-
 .../http_client/include/dap_http_client.h     |  17 +-
 .../http_client/include/dap_http_header.h     |  47 +--
 .../net/server/http_server/include/dap_http.h |   5 +
 .../http_server/include/dap_http_cache.h      |  40 +++
 .../http_server/include/dap_http_simple.h     |   2 +
 10 files changed, 411 insertions(+), 223 deletions(-)
 create mode 100644 dap-sdk/net/server/http_server/dap_http_cache.c
 create mode 100644 dap-sdk/net/server/http_server/include/dap_http_cache.h

diff --git a/dap-sdk/net/server/http_server/dap_http.c b/dap-sdk/net/server/http_server/dap_http.c
index d28c653537..04458bfb29 100644
--- a/dap-sdk/net/server/http_server/dap_http.c
+++ b/dap-sdk/net/server/http_server/dap_http.c
@@ -52,7 +52,7 @@
 #include "dap_http_header.h"
 #include "dap_http_client.h"
 
-#define LOG_TAG "dap_http"
+#define LOG_TAG "http"
 
 
 /**
@@ -61,18 +61,18 @@
  */
 int dap_http_init( )
 {
-  if ( dap_http_header_init() != 0 ) { // Init submodule for headers manipulations
-    log_it(L_CRITICAL,"Can't init HTTP headers processing submodule");
-    return -1;
-  }
-
-  if ( dap_http_client_init() !=0 ) { // Init submodule for HTTP client event processing
-    log_it(L_CRITICAL,"Can't init HTTP client submodule");
-    return -2;
-  }
-
-  log_it( L_NOTICE, "Initialized HTTP server module" );
-  return 0;
+    if ( dap_http_header_init() != 0 ) { // Init submodule for headers manipulations
+        log_it(L_CRITICAL,"Can't init HTTP headers processing submodule");
+        return -1;
+    }
+
+    if ( dap_http_client_init() !=0 ) { // Init submodule for HTTP client event processing
+        log_it(L_CRITICAL,"Can't init HTTP client submodule");
+        return -2;
+    }
+
+    log_it( L_NOTICE, "Initialized HTTP server module" );
+    return 0;
 }
 
 /**
@@ -80,32 +80,33 @@ int dap_http_init( )
  */
 void dap_http_deinit()
 {
-  dap_http_header_deinit( );
-  dap_http_client_deinit( );
+    dap_http_header_deinit( );
+    dap_http_client_deinit( );
 }
 
 
 /**
- * @brief dap_server_http_init Init HTTP server
- * @param sh Server instance
+ * @brief dap_server_http_init   Init HTTP server
+ * @param a_server               Server instance
+ * @param a_server_name          Server name
  * @return 0 if ok lesser number if error
  */
-int dap_http_new( dap_server_t *sh, const char * server_name )
+int dap_http_new( dap_server_t *a_server, const char * a_server_name )
 {
-  sh->_inheritor = calloc( 1, sizeof(dap_http_t) );
+    a_server->_inheritor = DAP_NEW_Z(dap_http_t);
 
-  dap_http_t *shttp = DAP_HTTP( sh );
+    dap_http_t *l_http = DAP_HTTP( a_server );
 
-  shttp->server = sh;
-  strncpy( shttp->server_name, server_name, sizeof(shttp->server_name)-1 );
+    l_http->server = a_server;
+    strncpy( l_http->server_name, a_server_name, sizeof(l_http->server_name)-1 );
 
-  sh->client_callbacks.new_callback    = dap_http_client_new;
-  sh->client_callbacks.delete_callback = dap_http_client_delete;
-  sh->client_callbacks.read_callback   = dap_http_client_read;
-  sh->client_callbacks.write_callback  = dap_http_client_write;
-  sh->client_callbacks.error_callback  = dap_http_client_error;
+    a_server->client_callbacks.new_callback    = dap_http_client_new;
+    a_server->client_callbacks.delete_callback = dap_http_client_delete;
+    a_server->client_callbacks.read_callback   = dap_http_client_read;
+    a_server->client_callbacks.write_callback  = dap_http_client_write;
+    a_server->client_callbacks.error_callback  = dap_http_client_error;
 
-  return 0;
+    return 0;
 }
 
 /**
@@ -113,59 +114,57 @@ int dap_http_new( dap_server_t *sh, const char * server_name )
  * @param sh Server's instance
  * @param arg Non-used argument
  */
-void dap_http_delete( dap_server_t *sh, void * arg )
+void dap_http_delete( dap_server_t *a_server, void * a_arg )
 {
-  (void) arg;
-  (void) sh;
-  dap_http_t *shttp = DAP_HTTP( sh );
-  dap_http_url_proc_t *up, *tmp;
-
-  HASH_ITER( hh, shttp->url_proc ,up, tmp ) {
-    HASH_DEL(shttp->url_proc, up);
-    if( up->_inheritor )
-      free( up->_inheritor );
-    free( up );
-  }
+    (void) a_arg;
+    dap_http_t *l_http = DAP_HTTP( a_server );
+    dap_http_url_proc_t *l_url_proc, *l_tmp;
+
+    HASH_ITER( hh, l_http->url_proc ,l_url_proc, l_tmp ) {
+        HASH_DEL(l_http->url_proc, l_url_proc);
+        if( l_url_proc->_inheritor )
+            DAP_DELETE(l_url_proc->_inheritor );
+        DAP_DELETE(l_url_proc );
+    }
 }
 
 
 /**
- * @brief dap_http_add_proc  Add custom procesor for the HTTP server
- * @param sh                Server's instance
- * @param url_path          Part of URL to be processed
- * @param read_callback     Callback for read in DATA state
- * @param write_callback    Callback for write in DATA state
- * @param error_callback    Callback for error processing
+ * @brief dap_http_add_proc   Add custom procesor for the HTTP server
+ * @param a_http              Server's instance
+ * @param a_url_path          Part of URL to be processed
+ * @param a_read_callback     Callback for read in DATA state
+ * @param a_write_callback    Callback for write in DATA state
+ * @param a_error_callback    Callback for error processing
  */
-void dap_http_add_proc(dap_http_t *sh, const char *url_path, void *internal
-                      ,dap_http_client_callback_t new_callback
-                      ,dap_http_client_callback_t delete_callback
-                      ,dap_http_client_callback_t headers_read_callback
-                      ,dap_http_client_callback_t headers_write_callback
-                      ,dap_http_client_callback_t data_read_callback
-                      ,dap_http_client_callback_t data_write_callback
-                      ,dap_http_client_callback_error_t error_callback
+void dap_http_add_proc(dap_http_t *a_http, const char *a_url_path, void *a_inheritor
+                      ,dap_http_client_callback_t a_new_callback
+                      ,dap_http_client_callback_t a_delete_callback
+                      ,dap_http_client_callback_t a_headers_read_callback
+                      ,dap_http_client_callback_t a_headers_write_callback
+                      ,dap_http_client_callback_t a_data_read_callback
+                      ,dap_http_client_callback_t a_data_write_callback
+                      ,dap_http_client_callback_error_t a_error_callback
 
                       )
 {
-  dap_http_url_proc_t *up = (dap_http_url_proc_t *) calloc( 1, sizeof(dap_http_url_proc_t) );
+    dap_http_url_proc_t *l_url_proc = DAP_NEW_Z(dap_http_url_proc_t);
 
-  strncpy( up->url, url_path, sizeof(up->url)-1 );
+    strncpy( l_url_proc->url, a_url_path, sizeof(l_url_proc->url)-1 );
 
-  up->new_callback    = new_callback;
-  up->delete_callback = delete_callback;
+    l_url_proc->new_callback    = a_new_callback;
+    l_url_proc->delete_callback = a_delete_callback;
 
-  up->data_read_callback = data_read_callback;
-  up->data_write_callback = data_write_callback;
-  up->headers_read_callback = headers_read_callback;
-  up->headers_write_callback = headers_write_callback;
-  up->error_callback = error_callback;
+    l_url_proc->data_read_callback = a_data_read_callback;
+    l_url_proc->data_write_callback = a_data_write_callback;
+    l_url_proc->headers_read_callback = a_headers_read_callback;
+    l_url_proc->headers_write_callback = a_headers_write_callback;
+    l_url_proc->error_callback = a_error_callback;
 
-  up->_inheritor = internal;
+    l_url_proc->_inheritor = a_inheritor;
 
-  HASH_ADD_STR( sh->url_proc, url, up );
+    HASH_ADD_STR( a_http->url_proc, url, l_url_proc );
 
-  log_it( L_DEBUG, "Added URL processor for '%s' path", up->url );
+    log_it( L_DEBUG, "Added URL processor for '%s' path", l_url_proc->url );
 }
 
-
diff --git a/dap-sdk/net/server/http_server/dap_http_cache.c b/dap-sdk/net/server/http_server/dap_http_cache.c
new file mode 100644
index 0000000000..5980b5e030
--- /dev/null
+++ b/dap-sdk/net/server/http_server/dap_http_cache.c
@@ -0,0 +1,69 @@
+/*
+ * Authors:
+ * Dmitriy A. Gearasimov <gerasimov.dmitriy@demlabs.net>
+ * DeM Labs Ltd.   https://demlabs.net
+ * Copyright  (c) 2021
+ * All rights reserved.
+
+ This file is part of DAP SDK the open source project
+
+    DAP SDK 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.
+
+    DAP SDK 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 any DAP SDK based project.  If not, see <http://www.gnu.org/licenses/>.
+*/
+#include "utlist.h"
+#include "dap_http.h"
+#include "dap_http_cache.h"
+
+#define LOG_TAG "http_cache"
+
+dap_http_cache_t * dap_http_cache_update(struct dap_http_url_proc * a_url_proc, byte_t * a_body, size_t a_body_size,
+                                         dap_http_header_t * a_headers, time_t a_ts_expire )
+{
+    dap_http_cache_t * l_ret = DAP_NEW_Z(dap_http_cache_t);
+    if(a_body_size){
+        l_ret->body = DAP_NEW_SIZE(byte_t,a_body_size);
+        memcpy(l_ret->body,a_body,a_body_size);
+        l_ret->body_size = a_body_size;
+    }
+    l_ret->headers =  dap_http_headers_dup( a_headers);
+    l_ret->ts_expire = a_ts_expire;
+    l_ret->url_proc = a_url_proc;
+    pthread_rwlock_wrlock(&a_url_proc->cache_rwlock);
+    dap_http_cache_delete(a_url_proc->cache);
+    a_url_proc->cache = l_ret;
+    pthread_rwlock_unlock(&a_url_proc->cache_rwlock);
+    return l_ret;
+}
+
+/**
+ * @brief dap_http_cache_delete
+ * @param a_http_cache
+ */
+void dap_http_cache_delete(dap_http_cache_t * a_http_cache)
+{
+   if (a_http_cache){
+       if(a_http_cache->body)
+           DAP_DELETE(a_http_cache->body);
+       dap_http_header_t *l_hdr=NULL, *l_tmp=NULL;
+
+       DL_FOREACH_SAFE(a_http_cache->headers,l_hdr,l_tmp){
+           DL_DELETE(a_http_cache->headers,l_hdr);
+           if(l_hdr->name)
+               DAP_DELETE(l_hdr->name);
+           if(l_hdr->value)
+               DAP_DELETE(l_hdr->value);
+           DAP_DELETE(l_hdr);
+       }
+       DAP_DELETE(a_http_cache);
+   }
+}
diff --git a/dap-sdk/net/server/http_server/dap_http_simple.c b/dap-sdk/net/server/http_server/dap_http_simple.c
index a159c2c4f5..1daea3f597 100644
--- a/dap-sdk/net/server/http_server/dap_http_simple.c
+++ b/dap-sdk/net/server/http_server/dap_http_simple.c
@@ -439,6 +439,19 @@ size_t dap_http_simple_reply(dap_http_simple_t *a_http_simple, void *a_data, siz
     return l_data_copy_size;
 }
 
+/**
+ * @brief dap_http_simple_make_cache_from_reply
+ * @param a_http_simple
+ * @param a_ts_expire
+ */
+dap_http_cache_t * dap_http_simple_make_cache_from_reply(dap_http_simple_t * a_http_simple, time_t a_ts_expire  )
+{
+    return dap_http_cache_update(a_http_simple->http_client->http->url_proc,
+                                 a_http_simple->reply_byte,
+                                 a_http_simple->reply_size,
+                                 a_http_simple->http_client->out_headers, a_ts_expire);
+}
+
 /**
  * @brief dap_http_simple_reply_f
  * @param shs
diff --git a/dap-sdk/net/server/http_server/http_client/dap_http_client.c b/dap-sdk/net/server/http_server/http_client/dap_http_client.c
index 64f1b00468..a3ae2a556c 100644
--- a/dap-sdk/net/server/http_server/http_client/dap_http_client.c
+++ b/dap-sdk/net/server/http_server/http_client/dap_http_client.c
@@ -46,6 +46,9 @@
 
 #define LOG_TAG "dap_http_client"
 
+static bool s_request_line_parse( dap_http_client_t *cl_ht, char *buf, size_t buf_length );
+
+
 /**
  * @brief dap_http_client_init Init HTTP client module
  * @return  Zero if ok others if not
@@ -66,50 +69,62 @@ void dap_http_client_deinit( )
 
 /**
  * @brief dap_http_client_new Creates HTTP client's internal structure
- * @param cl HTTP Client instance
- * @param arg Additional argument (usualy not used)
+ * @param a_esocket ESocket instance
+ * @param a_arg Additional argument (usualy not used)
  */
-void dap_http_client_new( dap_events_socket_t *cl, void *arg )
+void dap_http_client_new( dap_events_socket_t *a_esocket, void *a_arg )
 {
-    (void) arg;
-
+    (void) a_arg;
 
-    cl->_inheritor = DAP_NEW_Z( dap_http_client_t );
 
-    dap_http_client_t *cl_ht = DAP_HTTP_CLIENT( cl );
-    cl_ht->esocket = cl;
-    cl_ht->http = DAP_HTTP( cl->server );
-    cl_ht->state_read = DAP_HTTP_CLIENT_STATE_START;
-    cl_ht->state_write = DAP_HTTP_CLIENT_STATE_NONE;
+    a_esocket->_inheritor = DAP_NEW_Z( dap_http_client_t );
 
+    dap_http_client_t *l_http_client = DAP_HTTP_CLIENT( a_esocket );
+    l_http_client->esocket = a_esocket;
+    l_http_client->http = DAP_HTTP( a_esocket->server );
+    l_http_client->state_read = DAP_HTTP_CLIENT_STATE_START;
+    l_http_client->state_write = DAP_HTTP_CLIENT_STATE_NONE;
+
+    pthread_rwlock_rdlock(&l_http_client->http->url_proc->cache_rwlock);
+    if(l_http_client->http->url_proc->cache){
+        if (l_http_client->http->url_proc->cache->ts_expire < time(NULL) )
+            l_http_client->out_headers = dap_http_headers_dup(l_http_client->http->url_proc->cache->headers);
+        else{
+            pthread_rwlock_unlock(&l_http_client->http->url_proc->cache_rwlock);
+            pthread_rwlock_wrlock(&l_http_client->http->url_proc->cache_rwlock);
+            dap_http_cache_delete(l_http_client->http->url_proc->cache);
+            l_http_client->http->url_proc->cache = NULL;
+        }
+    }
+    pthread_rwlock_unlock(&l_http_client->http->url_proc->cache_rwlock);
     return;
 }
 
 /**
  * @brief dap_http_client_delete
- * @param cl HTTP Client instance
- * @param arg Additional argument (usualy not used)
+ * @param a_esocket HTTP Client instance's esocket
+ * @param a_arg Additional argument (usualy not used)
  */
-void dap_http_client_delete( dap_events_socket_t * cl, void *arg )
+void dap_http_client_delete( dap_events_socket_t * a_esocket, void *a_arg )
 {
-    dap_http_client_t *cl_ht = DAP_HTTP_CLIENT( cl );
-    if (cl_ht == NULL){ // Client is in proc callback in another thread so we don't delete it
+    (void) a_arg;
+    dap_http_client_t *l_http_client = DAP_HTTP_CLIENT( a_esocket );
+    if (l_http_client == NULL){ // Client is in proc callback in another thread so we don't delete it
         return;
     }
-    while( cl_ht->in_headers )
-        dap_http_header_remove( &cl_ht->in_headers, cl_ht->in_headers );
+    while( l_http_client->in_headers )
+        dap_http_header_remove( &l_http_client->in_headers, l_http_client->in_headers );
 
-    while( cl_ht->out_headers )
-        dap_http_header_remove( &cl_ht->out_headers, cl_ht->out_headers );
+    while( l_http_client->out_headers )
+        dap_http_header_remove( &l_http_client->out_headers, l_http_client->out_headers );
 
-    if( cl_ht->proc ) {
-        if( cl_ht->proc->delete_callback ) {
-          cl_ht->proc->delete_callback( cl_ht, NULL );
+    if( l_http_client->proc ) {
+        if( l_http_client->proc->delete_callback ) {
+          l_http_client->proc->delete_callback( l_http_client, NULL );
         }
     }
-    DAP_DEL_Z(cl_ht->_inheritor)
+    DAP_DEL_Z(l_http_client->_inheritor)
 
-  (void) arg;
 }
 
 
@@ -121,12 +136,12 @@ void dap_http_client_delete( dap_events_socket_t * cl, void *arg )
  */
 
 #if 1
-int detect_end_of_line( const char *buf, size_t max_size )
+static int detect_end_of_line( const char *a_buf, size_t a_max_size )
 {
   size_t i;
 
-  for( i = 0; i < max_size; i++ ) {
-    if ( buf[i] == '\n' ) {
+  for( i = 0; i < a_max_size; i++ ) {
+    if ( a_buf[i] == '\n' ) {
       return i;
     }
   }
@@ -135,7 +150,7 @@ int detect_end_of_line( const char *buf, size_t max_size )
 }
 #endif
 
-char  *z_basename( char *path, uint32_t len )
+static char  *z_basename( char *path, uint32_t len )
 {
   if ( !len )
     len = strlen( path );
@@ -156,7 +171,7 @@ char  *z_basename( char *path, uint32_t len )
   return ptr;
 }
 
-int32_t  z_dirname( char *path, uint32_t len )
+static int32_t  z_dirname( char *path, uint32_t len )
 {
   if ( !len )
     len = strlen( path );
@@ -180,7 +195,7 @@ int32_t  z_dirname( char *path, uint32_t len )
   return len;
 }
 
-int32_t  z_rootdirname( char *path, uint32_t len )
+static int32_t  z_rootdirname( char *path, uint32_t len )
 {
   if ( !len )
     len = strlen( path );
@@ -207,55 +222,55 @@ int32_t  z_rootdirname( char *path, uint32_t len )
 }
 
 /**
- * @brief dap_http_request_line_parse
- * @param cl_ht
- * @param buf
- * @param buf_length
+ * @brief s_request_line_parse
+ * @param a_http_client
+ * @param a_buf
+ * @param a_buf_length
  * @return
  */
-bool dap_http_request_line_parse( dap_http_client_t *cl_ht, char *buf, size_t buf_length )
+static bool s_request_line_parse( dap_http_client_t *a_http_client, char *a_buf, size_t a_buf_length )
 {
-  size_t pos;
-  size_t pos_kw_begin = 0;
+  size_t l_pos;
+  size_t l_pos_kw_begin = 0;
 
-  enum parse_state { PS_START = 0, PS_ACTION = 1, PS_URL = 2, PS_TYPE = 3, PS_VER_MAJOR = 4, PS_VER_MINOR = 5 }  p_st = PS_ACTION;
+  enum parse_state { PS_START = 0, PS_ACTION = 1, PS_URL = 2, PS_TYPE = 3, PS_VER_MAJOR = 4, PS_VER_MINOR = 5 }  l_parse_state = PS_ACTION;
 
   log_it( L_NOTICE, "dap_http_request_line_parse" );
 
-  for( pos = 0; pos < buf_length; pos ++ ) {
+  for( l_pos = 0; l_pos < a_buf_length; l_pos ++ ) {
 
-    if ( buf[pos] == '\n' )
+    if ( a_buf[l_pos] == '\n' )
       break;
 
-    if ( buf[pos] == ' ' || buf[pos] == '\t' ) {
+    if ( a_buf[l_pos] == ' ' || a_buf[l_pos] == '\t' ) {
 
-      switch( p_st ) {
+      switch( l_parse_state ) {
       case PS_ACTION:
       {
-        size_t c_size = pos - pos_kw_begin;
-        if ( c_size + 1 > sizeof(cl_ht->action) )
-          c_size = sizeof( cl_ht->action ) - 1;
+        size_t c_size = l_pos - l_pos_kw_begin;
+        if ( c_size + 1 > sizeof(a_http_client->action) )
+          c_size = sizeof( a_http_client->action ) - 1;
 
-        memcpy( cl_ht->action, buf + pos_kw_begin, c_size );
-        cl_ht->action[c_size] = 0;
-        log_it( L_WARNING, "Input: action '%s' pos=%u pos_kw_begin=%u", cl_ht->action, (uint32_t)pos, (uint32_t)pos_kw_begin );
+        memcpy( a_http_client->action, a_buf + l_pos_kw_begin, c_size );
+        a_http_client->action[c_size] = 0;
+        log_it( L_WARNING, "Input: action '%s' pos=%u pos_kw_begin=%u", a_http_client->action, (uint32_t)l_pos, (uint32_t)l_pos_kw_begin );
 
-        p_st = PS_URL;
-        pos_kw_begin = pos + 1;
+        l_parse_state = PS_URL;
+        l_pos_kw_begin = l_pos + 1;
       }
       break;
 
       case PS_URL:
       {
-        size_t c_size = pos - pos_kw_begin;
-        if ( c_size + 1 > sizeof(cl_ht->action) )
-          c_size = sizeof( cl_ht->url_path ) - 1;
-
-        memcpy( cl_ht->url_path, buf + pos_kw_begin, c_size );
-        cl_ht->url_path[c_size] = 0;
-        log_it( L_WARNING, "Input: url '%s' pos=%lu pos_kw_begin=%lu", cl_ht->url_path, (uint32_t)pos, (uint32_t)pos_kw_begin );
-        p_st = PS_TYPE;
-        pos_kw_begin = pos + 1;
+        size_t c_size = l_pos - l_pos_kw_begin;
+        if ( c_size + 1 > sizeof(a_http_client->action) )
+          c_size = sizeof( a_http_client->url_path ) - 1;
+
+        memcpy( a_http_client->url_path, a_buf + l_pos_kw_begin, c_size );
+        a_http_client->url_path[c_size] = 0;
+        log_it( L_WARNING, "Input: url '%s' pos=%lu pos_kw_begin=%lu", a_http_client->url_path, (uint32_t)l_pos, (uint32_t)l_pos_kw_begin );
+        l_parse_state = PS_TYPE;
+        l_pos_kw_begin = l_pos + 1;
         break;
       }
       break;
@@ -266,52 +281,52 @@ bool dap_http_request_line_parse( dap_http_client_t *cl_ht, char *buf, size_t bu
     }
   } // for
 
-  if ( pos_kw_begin < buf_length && p_st == PS_TYPE ) {
+  if ( l_pos_kw_begin < a_buf_length && l_parse_state == PS_TYPE ) {
 
-    size_t c_size;
+    size_t l_c_size;
 
-    char *end = memchr( buf + pos_kw_begin, '/', buf_length - pos_kw_begin );
+    char *end = memchr( a_buf + l_pos_kw_begin, '/', a_buf_length - l_pos_kw_begin );
 
-    if ( end && end < buf + buf_length ) {
+    if ( end && end < a_buf + a_buf_length ) {
 
-      c_size = end - (buf + pos_kw_begin);
+      l_c_size = end - (a_buf + l_pos_kw_begin);
       //TODO get version here
       //end = memchr( buf + pos_kw_begin, '/', buf_length - pos_kw_begin );
 
     }
     else
-      c_size = buf_length - pos_kw_begin;
+      l_c_size = a_buf_length - l_pos_kw_begin;
 
-    if ( c_size + 1 > sizeof(cl_ht->in_content_type) )
-       c_size = sizeof(cl_ht->in_content_type) - 1;
+    if ( l_c_size + 1 > sizeof(a_http_client->in_content_type) )
+       l_c_size = sizeof(a_http_client->in_content_type) - 1;
 
-    memcpy( cl_ht->in_content_type, buf + pos_kw_begin, c_size );
-    cl_ht->in_content_type[c_size] = 0;
+    memcpy( a_http_client->in_content_type, a_buf + l_pos_kw_begin, l_c_size );
+    a_http_client->in_content_type[l_c_size] = 0;
 
-    log_it( L_WARNING, "Input: type '%s' pos=%lu pos_kw_begin=%lu", cl_ht->in_content_type, (uint32_t)pos, (uint32_t)pos_kw_begin );
+    log_it( L_WARNING, "Input: type '%s' pos=%lu pos_kw_begin=%lu", a_http_client->in_content_type, (uint32_t)l_pos, (uint32_t)l_pos_kw_begin );
   }
 
-  return cl_ht->url_path[0] && cl_ht->action[0];
+  return a_http_client->url_path[0] && a_http_client->action[0];
 }
 
 /**
  * @brief s_report_error_and_restart
- * @param cl
- * @param cl_ht
+ * @param a_esocket
+ * @param a_http_client
  */
-static inline void s_report_error_and_restart( dap_events_socket_t *cl, dap_http_client_t *cl_ht )
+static inline void s_report_error_and_restart( dap_events_socket_t *a_esocket, dap_http_client_t *a_http_client )
 {
-  cl->buf_in_size = 0;
-  cl_ht->state_read = DAP_HTTP_CLIENT_STATE_NONE;
+    a_esocket->buf_in_size = 0;
+    a_http_client->state_read = DAP_HTTP_CLIENT_STATE_NONE;
 
-  dap_events_socket_set_readable_unsafe( cl_ht->esocket, false );
-  dap_events_socket_set_writable_unsafe( cl_ht->esocket, true );
+    dap_events_socket_set_readable_unsafe( a_http_client->esocket, false );
+    dap_events_socket_set_writable_unsafe( a_http_client->esocket, true );
 
-  cl_ht->reply_status_code = 505;
-  strcpy( cl_ht->reply_reason_phrase, "Error" );
-  cl_ht->state_write = DAP_HTTP_CLIENT_STATE_START;
+    a_http_client->reply_status_code = 505;
+    strcpy( a_http_client->reply_reason_phrase, "Error" );
+    a_http_client->state_write = DAP_HTTP_CLIENT_STATE_START;
 
-  return;
+    return;
 }
 
 /**
@@ -319,7 +334,7 @@ static inline void s_report_error_and_restart( dap_events_socket_t *cl, dap_http
  * @param cl HTTP Client instance
  * @param arg Additional argument (usualy not used)
  */
-void dap_http_client_read( dap_events_socket_t *a_esocket, void *arg )
+void dap_http_client_read( dap_events_socket_t *a_esocket, void *a_arg )
 {
     dap_http_client_t *l_http_client = DAP_HTTP_CLIENT( a_esocket );
 
@@ -358,7 +373,7 @@ void dap_http_client_read( dap_events_socket_t *a_esocket, void *arg )
                 l_buf_line[ eol + 2 ] = 0; // null terminate
 
                 // parse http_request_line
-                if ( !dap_http_request_line_parse(l_http_client, l_buf_line, eol + 1) ) {
+                if ( !s_request_line_parse(l_http_client, l_buf_line, eol + 1) ) {
                     log_it( L_WARNING, "Input: Wrong request line '%s'", l_buf_line );
                     s_report_error_and_restart( a_esocket, l_http_client );
                     break;
@@ -474,30 +489,41 @@ void dap_http_client_read( dap_events_socket_t *a_esocket, void *arg )
 
 /**
  * @brief dap_http_client_write Process write event
- * @param cl HTTP Client instance
- * @param arg Additional argument (usualy not used)
+ * @param a_esocket HTTP Client instance's esocket
+ * @param a_arg Additional argument (usualy not used)
  */
-void dap_http_client_write( dap_events_socket_t * cl, void *arg )
+void dap_http_client_write( dap_events_socket_t * a_esocket, void *a_arg )
 {
     //  log_it( L_DEBUG, "dap_http_client_write..." );
 
-    (void) arg;
-    dap_http_client_t *l_http_client = DAP_HTTP_CLIENT( cl );
+    (void) a_arg;
+    dap_http_client_t *l_http_client = DAP_HTTP_CLIENT( a_esocket );
     //log_it(L_WARNING,"HTTP client write callback in state %d",l_http_client->state_write);
 
     switch( l_http_client->state_write ) {
         case DAP_HTTP_CLIENT_STATE_NONE:
             return;
         case DAP_HTTP_CLIENT_STATE_START:{
-            if ( l_http_client->proc )
-                if ( l_http_client->proc->headers_write_callback )
-                    l_http_client->proc->headers_write_callback( l_http_client, NULL );
+            if ( l_http_client->proc ){
+                // We check out_headers because if they are - we send only cached headers and don't call headers_write_callback at all
+                if ( l_http_client->out_headers==NULL && l_http_client->proc->headers_write_callback ){
+                        l_http_client->proc->headers_write_callback( l_http_client, NULL );
+                        dap_http_client_out_header_generate( l_http_client );
+                }else if (l_http_client->out_headers){
+                    l_http_client->reply_status_code = Http_Status_OK; // Cached data are always OK... for now.
+                    //TODO: make cached reply status code
+                }
+            }
+            char buf[1024];
+            time_t current_time = time( NULL );
+            dap_time_to_str_rfc822( buf, sizeof(buf), current_time );
+
+            dap_http_header_add( &l_http_client->out_headers,"Date", buf );
 
             log_it( L_INFO," HTTP response with %u status code", l_http_client->reply_status_code );
-            dap_events_socket_write_f_unsafe(cl, "HTTP/1.1 %u %s\r\n",l_http_client->reply_status_code, l_http_client->reply_reason_phrase[0] ?
+            dap_events_socket_write_f_unsafe(a_esocket, "HTTP/1.1 %u %s\r\n",l_http_client->reply_status_code, l_http_client->reply_reason_phrase[0] ?
                             l_http_client->reply_reason_phrase : http_status_reason_phrase(l_http_client->reply_status_code) );
-            dap_events_socket_set_writable_unsafe(cl, true);
-            dap_http_client_out_header_generate( l_http_client );
+            dap_events_socket_set_writable_unsafe(a_esocket, true);
             l_http_client->state_write = DAP_HTTP_CLIENT_STATE_HEADERS;
         } break;
 
@@ -506,33 +532,48 @@ void dap_http_client_write( dap_events_socket_t * cl, void *arg )
             if ( hdr == NULL ) {
                 log_it(L_DEBUG, "Output: headers are over (reply status code %u content_lentgh %u)",
                        l_http_client->reply_status_code, l_http_client->out_content_length);
-                dap_events_socket_write_f_unsafe(cl, "\r\n");
-                dap_events_socket_set_writable_unsafe(cl, true);
+                dap_events_socket_write_f_unsafe(a_esocket, "\r\n");
+                dap_events_socket_set_writable_unsafe(a_esocket, true);
                 if ( l_http_client->out_content_length || l_http_client->out_content_ready ) {
                     l_http_client->state_write=DAP_HTTP_CLIENT_STATE_DATA;
                 } else {
                     log_it( L_DEBUG, "Nothing to output" );
                     l_http_client->state_write = DAP_HTTP_CLIENT_STATE_NONE;
-                    dap_events_socket_set_writable_unsafe( cl, false );
-                    cl->flags |= DAP_SOCK_SIGNAL_CLOSE;
+                    dap_events_socket_set_writable_unsafe( a_esocket, false );
+                    a_esocket->flags |= DAP_SOCK_SIGNAL_CLOSE;
                 }
-                dap_events_socket_set_readable_unsafe( cl, true );
+                dap_events_socket_set_readable_unsafe( a_esocket, true );
             } else {
                 //log_it(L_WARNING,"Output: header %s: %s",hdr->name,hdr->value);
-                dap_events_socket_write_f_unsafe(cl, "%s: %s\r\n", hdr->name, hdr->value);
-                dap_events_socket_set_writable_unsafe(cl, true);
+                dap_events_socket_write_f_unsafe(a_esocket, "%s: %s\r\n", hdr->name, hdr->value);
+                dap_events_socket_set_writable_unsafe(a_esocket, true);
                 dap_http_header_remove( &l_http_client->out_headers, hdr );
             }
         } break;
-        case DAP_HTTP_CLIENT_STATE_DATA:
-        {
-          if ( l_http_client->proc ){
-            if ( l_http_client->proc->data_write_callback ){
-                l_http_client->proc->data_write_callback( l_http_client, NULL );
+        case DAP_HTTP_CLIENT_STATE_DATA:{
+            if ( l_http_client->proc ){
+                pthread_rwlock_rdlock(&l_http_client->proc->cache_rwlock);
+                if ( l_http_client->proc->cache == NULL && l_http_client->proc->data_write_callback ){
+                    pthread_rwlock_unlock(&l_http_client->proc->cache_rwlock);
+                    l_http_client->proc->data_write_callback( l_http_client, NULL );
+                }else if(l_http_client->proc->cache) {
+                    size_t l_to_send=l_http_client->proc->cache->body_size-l_http_client->out_cache_position ;
+                    size_t l_sent = dap_events_socket_write_unsafe(l_http_client->esocket,
+                                                   l_http_client->proc->cache->body+l_http_client->out_cache_position,
+                                                   l_to_send );
+                    if(l_sent){
+                        if ( l_http_client->out_cache_position + l_sent >= l_http_client->proc->cache->body_size ){ // All is sent
+                            l_http_client->esocket->flags |= DAP_SOCK_SIGNAL_CLOSE;
+                            l_http_client->state_write = DAP_HTTP_CLIENT_STATE_NONE;
+                            dap_events_socket_set_writable_unsafe( a_esocket, false );
+                        }else
+                            l_http_client->out_cache_position += l_sent;
+                    }
+                    pthread_rwlock_unlock(&l_http_client->proc->cache_rwlock);
+                }
+            }else{
+                log_it(L_WARNING, "No http proc, nothing to write");
             }
-          }else{
-              log_it(L_WARNING, "No http proc, nothing to write");
-          }
         }
         break;
   }
@@ -545,10 +586,6 @@ void dap_http_client_write( dap_events_socket_t * cl, void *arg )
 void dap_http_client_out_header_generate(dap_http_client_t *cl_ht)
 {
   char buf[1024];
-  time_t current_time = time( NULL );
-  dap_time_to_str_rfc822( buf, sizeof(buf), current_time );
-
-  dap_http_header_add( &cl_ht->out_headers,"Date", buf );
 
   if ( cl_ht->reply_status_code == 200 ) {
 
diff --git a/dap-sdk/net/server/http_server/http_client/dap_http_header.c b/dap-sdk/net/server/http_server/http_client/dap_http_header.c
index e2458cadef..59ee805194 100644
--- a/dap-sdk/net/server/http_server/http_client/dap_http_header.c
+++ b/dap-sdk/net/server/http_server/http_client/dap_http_header.c
@@ -1,24 +1,26 @@
 /*
- Copyright (c) 2017-2018 (c) Project "DeM Labs Inc" https://github.com/demlabsinc
-  All rights reserved.
+ * Authors:
+ * Dmitriy A. Gearasimov <gerasimov.dmitriy@demlabs.net>
+ * DeM Labs Ltd.   https://demlabs.net
+ * Copyright  (c) 2021
+ * All rights reserved.
 
- This file is part of DAP (Deus Applications Prototypes) the open source project
+ This file is part of DAP SDK the open source project
 
-    DAP (Deus Applicaions Prototypes) is free software: you can redistribute it and/or modify
-    it under the terms of the GNU Lesser General Public License as published by
+    DAP SDK 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.
 
-    DAP is distributed in the hope that it will be useful,
+    DAP SDK 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 Lesser General Public License for more details.
+    GNU General Public License for more details.
 
-    You should have received a copy of the GNU Lesser General Public License
-    along with any DAP based project.  If not, see <http://www.gnu.org/licenses/>.
+    You should have received a copy of the GNU General Public License
+    along with any DAP SDK based project.  If not, see <http://www.gnu.org/licenses/>.
 */
 
-
 #include <stdio.h>
 #include <stdarg.h>
 #include <string.h>
@@ -32,8 +34,10 @@
 #endif
 
 #include <pthread.h>
+#include <utlist.h>
 
 #include "dap_common.h"
+#include "dap_strfuncs.h"
 #include "dap_events_socket.h"
 #include "dap_http_client.h"
 #include "dap_http_header.h"
@@ -215,3 +219,20 @@ dap_http_header_t *dap_http_header_find( dap_http_header_t *top, const char *nam
 
   return ret;
 }
+
+/**
+ * @brief dap_http_headers_dup
+ * @param a_top
+ * @return
+ */
+dap_http_header_t * dap_http_headers_dup(dap_http_header_t * a_top)
+{
+    dap_http_header_t * l_hdr=NULL, * l_ret = NULL;
+    DL_FOREACH(a_top,l_hdr){
+        dap_http_header_t * l_hdr_copy = DAP_NEW_Z(dap_http_header_t);
+        l_hdr_copy->name = dap_strdup(l_hdr->name);
+        l_hdr_copy->value = dap_strdup(l_hdr->value);
+        DL_APPEND(l_ret,l_hdr_copy);
+    }
+    return l_ret;
+}
diff --git a/dap-sdk/net/server/http_server/http_client/include/dap_http_client.h b/dap-sdk/net/server/http_server/http_client/include/dap_http_client.h
index 864233216d..2be93454e1 100644
--- a/dap-sdk/net/server/http_server/http_client/include/dap_http_client.h
+++ b/dap-sdk/net/server/http_server/http_client/include/dap_http_client.h
@@ -62,6 +62,7 @@ typedef struct dap_http_client
     char out_content_type[256];
     time_t out_last_modified;
     bool out_connection_close;
+    size_t out_cache_position;
 
     dap_events_socket_t *esocket;
     struct dap_http * http;
@@ -82,15 +83,15 @@ typedef struct dap_http_client
 extern "C" {
 #endif
 
-int dap_http_client_init( );
-void dap_http_client_deinit( );
-void dap_http_client_new( dap_events_socket_t * cl,void *arg ); // Creates HTTP client's internal structure
-void dap_http_client_delete( dap_events_socket_t * cl,void *arg ); // Free memory for HTTP client's internal structure
+int dap_http_client_init(void);
+void dap_http_client_deinit(void);
+void dap_http_client_new( dap_events_socket_t *a_esocket, void *a_arg ); // Creates HTTP client's internal structure
+void dap_http_client_delete( dap_events_socket_t * a_esocket,void *a_arg ); // Free memory for HTTP client's internal structure
 
-void dap_http_client_read( dap_events_socket_t * cl,void *arg ); // Process read event
-void dap_http_client_write( dap_events_socket_t * cl,void *arg ); // Process write event
-void dap_http_client_error( dap_events_socket_t * cl,int arg ); // Process error event
-void dap_http_client_out_header_generate( dap_http_client_t *cl_ht );
+void dap_http_client_read( dap_events_socket_t * a_esocket,void *a_arg ); // Process read event
+void dap_http_client_write( dap_events_socket_t * a_esocket,void *a_arg ); // Process write event
+void dap_http_client_error( dap_events_socket_t * a_esocket,int a_arg ); // Process error event
+void dap_http_client_out_header_generate( dap_http_client_t *a_http_client );
 
 #ifdef __cplusplus
 }
diff --git a/dap-sdk/net/server/http_server/http_client/include/dap_http_header.h b/dap-sdk/net/server/http_server/http_client/include/dap_http_header.h
index 55c3e37054..f3b1e8e284 100644
--- a/dap-sdk/net/server/http_server/http_client/include/dap_http_header.h
+++ b/dap-sdk/net/server/http_server/http_client/include/dap_http_header.h
@@ -1,27 +1,27 @@
 /*
- Copyright (c) 2017-2018 (c) Project "DeM Labs Inc" https://github.com/demlabsinc
-  All rights reserved.
+ * Authors:
+ * Dmitriy A. Gearasimov <gerasimov.dmitriy@demlabs.net>
+ * DeM Labs Ltd.   https://demlabs.net
+ * Copyright  (c) 2021
+ * All rights reserved.
 
- This file is part of DAP (Deus Applications Prototypes) the open source project
+ This file is part of DAP SDK the open source project
 
-    DAP (Deus Applicaions Prototypes) is free software: you can redistribute it and/or modify
-    it under the terms of the GNU Lesser General Public License as published by
+    DAP SDK 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.
 
-    DAP is distributed in the hope that it will be useful,
+    DAP SDK 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 Lesser General Public License for more details.
+    GNU General Public License for more details.
 
-    You should have received a copy of the GNU Lesser General Public License
-    along with any DAP based project.  If not, see <http://www.gnu.org/licenses/>.
+    You should have received a copy of the GNU General Public License
+    along with any DAP SDK based project.  If not, see <http://www.gnu.org/licenses/>.
 */
 
-
-#ifndef _DAP_HTTP_HEADER_H_
-#define _DAP_HTTP_HEADER_H_
-
+#pragma once
 //Structure for holding HTTP header in the bidirectional list
 typedef struct dap_http_header{
     char *name;
@@ -32,21 +32,22 @@ typedef struct dap_http_header{
 
 struct dap_http_client;
 
-extern int dap_http_header_init(); // Init module
-extern void dap_http_header_deinit(); // Deinit module
+int dap_http_header_init(); // Init module
+void dap_http_header_deinit(); // Deinit module
+
+int dap_http_header_parse(struct dap_http_client * cl_ht, const char * str);
 
-extern int dap_http_header_parse(struct dap_http_client * cl_ht, const char * str);
+dap_http_header_t * dap_http_header_add(dap_http_header_t ** top, const char*name, const char * value);
 
-extern dap_http_header_t * dap_http_header_add(dap_http_header_t ** top, const char*name, const char * value);
+dap_http_header_t * dap_http_out_header_add(struct dap_http_client * ht, const char*name, const char * value);
+dap_http_header_t * dap_http_out_header_add_f(struct dap_http_client * ht, const char*name, const char * value,...);
 
-extern dap_http_header_t * dap_http_out_header_add(struct dap_http_client * ht, const char*name, const char * value);
-extern dap_http_header_t * dap_http_out_header_add_f(struct dap_http_client * ht, const char*name, const char * value,...);
+dap_http_header_t * dap_http_header_find(dap_http_header_t * top, const char*name);
 
-extern dap_http_header_t * dap_http_header_find(dap_http_header_t * top, const char*name);
+dap_http_header_t * dap_http_headers_dup(dap_http_header_t * a_top);
 
-extern void dap_http_header_remove(dap_http_header_t ** top,dap_http_header_t * hdr );
+void dap_http_header_remove(dap_http_header_t ** top,dap_http_header_t * hdr );
 
 // For debug output
-extern void print_dap_http_headers(dap_http_header_t * top);
+void print_dap_http_headers(dap_http_header_t * top);
 
-#endif
diff --git a/dap-sdk/net/server/http_server/include/dap_http.h b/dap-sdk/net/server/http_server/include/dap_http.h
index 0d1fcfb84f..257b73e009 100644
--- a/dap-sdk/net/server/http_server/include/dap_http.h
+++ b/dap-sdk/net/server/http_server/include/dap_http.h
@@ -26,6 +26,7 @@ See more details here <http://www.gnu.org/licenses/>.
 #include "dap_events_socket.h"
 #include "dap_http_header.h"
 #include "dap_http_client.h"
+#include "dap_http_cache.h"
 #include "uthash.h"
 
 struct dap_http;
@@ -37,6 +38,9 @@ typedef struct dap_http_url_proc{
     char url[512]; // First part of URL that will be processed
     struct dap_http * http; // Pointer to HTTP server instance
 
+    dap_http_cache_t * cache; // In memory cache, could be present or not
+    pthread_rwlock_t cache_rwlock;
+
     dap_http_client_callback_t new_callback; // Init internal structure
     dap_http_client_callback_t delete_callback; // Delete internal structure
 
@@ -77,3 +81,4 @@ void dap_http_add_proc(dap_http_t *sh, const char *url_path, void *internal
                              ,dap_http_client_callback_t data_write_callback
                              ,dap_http_client_callback_error_t error_callback ); // Add custom procesor for the HTTP server
 
+void dap_http_url_proc_cache_reset(dap_http_url_proc_t *a_url_proc);
diff --git a/dap-sdk/net/server/http_server/include/dap_http_cache.h b/dap-sdk/net/server/http_server/include/dap_http_cache.h
new file mode 100644
index 0000000000..c1e55a8d0a
--- /dev/null
+++ b/dap-sdk/net/server/http_server/include/dap_http_cache.h
@@ -0,0 +1,40 @@
+/*
+ * Authors:
+ * Dmitriy A. Gearasimov <gerasimov.dmitriy@demlabs.net>
+ * DeM Labs Ltd.   https://demlabs.net
+ * Copyright  (c) 2021
+ * All rights reserved.
+
+ This file is part of DAP SDK the open source project
+
+    DAP SDK 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.
+
+    DAP SDK 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 any DAP SDK based project.  If not, see <http://www.gnu.org/licenses/>.
+*/
+#pragma once
+#include "dap_common.h"
+#include "dap_http_header.h"
+
+// Cache object
+typedef struct dap_http_cache
+{
+    struct dap_http_url_proc * url_proc;
+    byte_t *body;
+    size_t body_size;
+    dap_http_header_t * headers;
+
+    time_t ts_expire;
+} dap_http_cache_t;
+
+dap_http_cache_t * dap_http_cache_update(struct dap_http_url_proc * a_url_proc, byte_t * a_body, size_t a_body_size,
+                                         dap_http_header_t * a_headers, time_t ts_expire );
+void dap_http_cache_delete(dap_http_cache_t * a_http_cache);
diff --git a/dap-sdk/net/server/http_server/include/dap_http_simple.h b/dap-sdk/net/server/http_server/include/dap_http_simple.h
index bd2ccc1a51..a0d1534035 100644
--- a/dap-sdk/net/server/http_server/include/dap_http_simple.h
+++ b/dap-sdk/net/server/http_server/include/dap_http_simple.h
@@ -65,6 +65,7 @@ typedef struct dap_http_simple {
 #define DAP_HTTP_SIMPLE(a) ((dap_http_simple_t*) (a)->_inheritor )
 
 void dap_http_simple_proc_add( dap_http_t *sh, const char *url_path, size_t reply_size_max, dap_http_simple_callback_t cb ); // Add simple processor
+
 int  dap_http_simple_module_init( void );
 void dap_http_simple_module_deinit(void);
 
@@ -83,4 +84,5 @@ void dap_http_simple_set_pass_unknown_user_agents( bool pass );
 
 size_t dap_http_simple_reply( dap_http_simple_t *shs, void *data, size_t data_size );
 size_t dap_http_simple_reply_f( dap_http_simple_t *shs, const char *data, ... );
+dap_http_cache_t * dap_http_simple_make_cache_from_reply(dap_http_simple_t * a_http_simple , time_t a_ts_expire );
 
-- 
GitLab