MEDIUM: lua: Improve/revisit the lua api to manipulate channels

First of all, following functions are now considered deprecated:

  * Channel:dup()
  * Channel:get()
  * Channel:getline()
  * Channel:get_in_len()
  * Cahnnel:get_out_len()

It is just informative, there is no warning and functions may still be
used. Howver it is recommended to use new functions. New functions are more
flexible and use a better naming pattern. In addition, the same names will
be used in the http_msg class to manipulate http messages from lua filters.

The new API is:

  * Channel:data()
  * Channel:line()
  * Channel:append()
  * Channel:prepend()
  * Channel:insert()
  * Channel:remove()
  * Channel:set()
  * Channel:input()
  * Channel:output()
  * Channel:send()
  * Channel:forward()
  * Channel:is_resp()
  * Channel:is_full()
  * Channel:may_recv()

The lua documentation was updated accordingly.
diff --git a/doc/lua-api/index.rst b/doc/lua-api/index.rst
index d98c77e..06468eb 100644
--- a/doc/lua-api/index.rst
+++ b/doc/lua-api/index.rst
@@ -1278,106 +1278,304 @@
 
   The following diagram shows where the channel class function are applied.
 
-  **Warning**: It is not possible to read from the response in request action,
-  and it is not possible to read for the request channel in response action.
+  .. image:: _static/channel.png
 
-  **Warning**: It is forbidden to alter the Channels buffer from HTTP contexts.
-  So only :js:func:`Channel.get_in_length`, :js:func:`Channel.get_out_length`
-  and :js:func:`Channel.is_full` can be called from an HTTP conetext.
+  .. warning::
+    It is not possible to read from the response in request action, and it is
+    not possible to read for the request channel in response action.
 
-.. image:: _static/channel.png
+  .. warning::
+    It is forbidden to alter the Channels buffer from HTTP contexts.  So only
+    :js:func:`Channel.input`, :js:func:`Channel.output`,
+    :js:func:`Channel.may_recv`, :js:func:`Channel.is_full` and
+    :js:func:`Channel.is_resp` can be called from an HTTP conetext.
 
-.. js:function:: Channel.dup(channel)
+  .. note::
+    Channel object may only be manipulated in the context of a registered
+    action, sample-fetch or converter. However, only actions are allowed to
+    yield. When it is said one of the following functions may yield, it is only
+    true in the context of an action.
+
+.. js:function:: Channel.append(channel, string)
 
-  This function returns a string that contain the entire buffer. The data is
-  not remove from the buffer and can be reprocessed later.
+  This function copies the string **string** at the end of incoming data of the
+  channel buffer. The function returns the copied length on success or -1 if
+  data cannot be copied.
 
-  If the buffer can't receive more data, a 'nil' value is returned.
+  Same that :js:func:`Channel.insert(channel, string, channel:input())`.
 
   :param class_channel channel: The manipulated Channel.
-  :returns: a string containing all the available data or nil.
+  :param string string: The data to copied into incoming data.
+  :returns: an integer containing the amount of bytes copied or -1.
 
-.. js:function:: Channel.get(channel)
+.. js:function:: Channel.data(channel [, offset [, length]])
 
-  This function returns a string that contain the entire buffer. The data is
-  consumed from the buffer.
+  This function returns **length** bytes of incoming data from the channel
+  buffer, starting at the offset **offset**. The data are not removed from the
+  buffer.
 
-  If the buffer can't receive more data, a 'nil' value is returned.
+  By default, if no length is provided, all incoming data found, starting at the
+  given offset, are returned. If **length** is set to -1, the function tries to
+  retrieve a maximum of data and yields if necessary. It also waits for more
+  data if the requested length exceeds the available amount of incoming data. Do
+  not providing an offset is the same that setting it to 0. A positive offset is
+  relative to the begining of incoming data of the channel buffer while negative
+  offset is relative to their end.
+
+  If there is no incoming data and the channel can't receive more data, a 'nil'
+  value is returned.
 
   :param class_channel channel: The manipulated Channel.
-  :returns: a string containing all the available data or nil.
+  :param integer offset: *optional* The offset in incoming data to start to get
+                         data. 0 by default. May be negative to be relative to
+                         the end of incoming data.
+  :param integer length: *optional* The expected length of data to retrieve. All
+                         incoming data by default. May be set to -1 to get a
+                         maximum of data.
+  :returns: a string containing the data found or nil.
 
-.. js:function:: Channel.getline(channel)
+.. js:function:: Channel.forward(channel, length)
 
-  This function returns a string that contain the first line of the buffer. The
-  data is consumed. If the data returned doesn't contains a final '\n' its
-  assumed than its the last available data in the buffer.
+  This function forwards **length** bytes of data from the channel buffer. If
+  the requested length exceeds the available amount of incoming data, the
+  function yields, waiting for more data to forward. It returns the amount of
+  data forwarded.
+
+  :param class_channel channel: The manipulated Channel.
+  :param integer int: The amount of data to forward.
 
-  If the buffer can't receive more data, a 'nil' value is returned.
+.. js:function:: Channel.input(channel)
+
+  This function returns the length of incoming data of the channel buffer.
 
   :param class_channel channel: The manipulated Channel.
-  :returns: a string containing the available line or nil.
+  :returns: an integer containing the amount of available bytes.
 
-.. js:function:: Channel.set(channel, string)
+.. js:function:: Channel.insert(channel, string [, offset])
 
-  This function replace the content of the buffer by the string. The function
-  returns the copied length, otherwise, it returns -1.
+  This function copies the string **string** at the offset **offset** in
+  incoming data of the channel buffer. The function returns the copied length on
+  success or -1 if data cannot be copied.
 
-  The data set with this function are not send. They wait for the end of
-  HAProxy processing, so the buffer can be full.
+  By default, if no offset is provided, the string is copied in front of
+  incoming data. A positive offset is relative to the begining of incoming data
+  of the channel buffer while negative offset is relative to their end.
 
   :param class_channel channel: The manipulated Channel.
-  :param string string: The data which will sent.
+  :param string string: The data to copied into incoming data.
+  :param integer offset: *optional* The offset in incomding data where to copied
+                         data. 0 by default. May be negative to be relative to
+                         the end of incoming data.
   :returns: an integer containing the amount of bytes copied or -1.
 
-.. js:function:: Channel.append(channel, string)
+.. js:function:: Channel.is_full(channel)
+
+  This function returns true if the channel buffer is full.
+
+  :returns: a boolean
+
+.. js:function:: Channel.is_resp(channel)
+
+  This function returns true if the channel is the response one.
+
+  :returns: a boolean
+
+.. js:function:: Channel.line(channel [, offset [, length]])
 
-  This function append the string argument to the content of the buffer. The
-  function returns the copied length, otherwise, it returns -1.
+  This function parses **length** bytes of incoming data of the channel buffer,
+  starting at offset **offset**, and returns the first line found, including the
+  '\n'.  The data are not removed from the buffer. If no line is found, all data
+  are returned.
 
-  The data set with this function are not send. They wait for the end of
-  HAProxy processing, so the buffer can be full.
+  By default, if no length is provided, all incoming data, starting at the given
+  offset, are evaluated. If **length** is set to -1, the function tries to
+  retrieve a maximum of data and yields if necessary. It also waits for more
+  data if the requested length exceeds the available amount of incoming data. Do
+  not providing an offset is the same that setting it to 0. A positive offset is
+  relative to the begining of incoming data of the channel buffer while negative
+  offset is relative to their end.
+
+  If there is no incoming data and the channel can't receive more data, a 'nil'
+  value is returned.
 
   :param class_channel channel: The manipulated Channel.
-  :param string string: The data which will sent.
+  :param integer offset: *optional* The offset in incomding data to start to
+                         parse data. 0 by default. May be negative to be
+                         relative to the end of incoming data.
+  :param integer length: *optional* The length of data to parse. All incoming
+                         data by default. May be set to -1 to get a maximum of
+                         data.
+  :returns: a string containing the line found or nil.
+
+.. js:function:: Channel.may_recv(channel)
+
+  This function returns true if the channel may still receive data.
+
+  :returns: a boolean
+
+.. js:function:: Channel.output(channel)
+
+  This function returns the length of outgoing data of the channel buffer.
+
+  :param class_channel channel: The manipulated Channel.
+  :returns: an integer containing the amount of available bytes.
+
+.. js:function:: Channel.prepend(channel, string)
+
+  This function copies the string **string** in front of incoming data of the
+  channel buffer. The function returns the copied length on success or -1 if
+  data cannot be copied.
+
+  Same that :js:func:`Channel.insert(channel, string, 0)`.
+
+  :param class_channel channel: The manipulated Channel.
+  :param string string: The data to copied into incoming data.
   :returns: an integer containing the amount of bytes copied or -1.
 
+.. js:function:: Channel.remove(channel [, offset [, length]])
+
+  This function removes **length** bytes of incoming data of the channel buffer,
+  starting at offset **offset**. This function returns number of bytes removed
+  on success.
+
+  By default, if no length is provided, all incoming data, starting at the given
+  offset, are removed. Do not providing an offset is the same that setting it
+  to 0. A positive offset is relative to the begining of incoming data of the
+  channel buffer while negative offset is relative to their end.
+
+  :param class_channel channel: The manipulated Channel.
+  :param integer offset: *optional* The offset in incomding data where to start
+                         to remove data. 0 by default. May be negative to
+                         be relative to the end of incoming data.
+  :param integer length: *optional* The length of data to remove. All incoming
+                         data by default.
+  :returns: an integer containing the amount of bytes removed.
+
 .. js:function:: Channel.send(channel, string)
 
-  This function required immediate send of the data. Unless if the connection
-  is close, the buffer is regularly flushed and all the string can be sent.
+  This function required immediate send of the string **string**. It means the
+  string is copied at the begining of incoming data of the channel buffer and
+  immediately forwarded. Unless if the connection is close, this function yields
+  to copied and forward all the string.
 
   :param class_channel channel: The manipulated Channel.
-  :param string string: The data which will sent.
+  :param string string: The data to send.
   :returns: an integer containing the amount of bytes copied or -1.
 
-.. js:function:: Channel.get_in_length(channel)
+.. js:function:: Channel.set(channel, string [, offset [, length]])
 
-  This function returns the length of the input part of the buffer.
+  This function replace **length** bytes of incoming data of the channel buffer,
+  starting at offset **offset**, by the string **string**. The function returns
+  the copied length on success or -1 if data cannot be copied.
+
+  By default, if no length is provided, all incoming data, starting at the given
+  offset, are replaced. Do not providing an offset is the same that setting it
+  to 0. A positive offset is relative to the begining of incoming data of the
+  channel buffer while negative offset is relative to their end.
 
   :param class_channel channel: The manipulated Channel.
-  :returns: an integer containing the amount of available bytes.
+  :param string string: The data to copied into incoming data.
+  :param integer offset: *optional* The offset in incomding data where to start
+                         the data replacement. 0 by default. May be negative to
+                         be relative to the end of incoming data.
+  :param integer length: *optional* The length of data to replace. All incoming
+                         data by default.
+  :returns: an integer containing the amount of bytes copied or -1.
 
-.. js:function:: Channel.get_out_length(channel)
+.. js:function:: Channel.dup(channel)
 
-  This function returns the length of the output part of the buffer.
+  **DEPRECATED**
+
+  This function returns all incoming data found in the channel buffer. The data
+  are not remove from the buffer and can be reprocessed later.
+
+  If there is no incoming data and the channel can't receive more data, a 'nil'
+  value is returned.
 
   :param class_channel channel: The manipulated Channel.
-  :returns: an integer containing the amount of available bytes.
+  :returns: a string containing all data found or nil.
+
+  .. warning::
+     This function is deprecated. :js:func:`Channel.data()` must be used
+     instead.
 
-.. js:function:: Channel.forward(channel, int)
+.. js:function:: Channel.get(channel)
+
+  **DEPRECATED**
 
-  This function transfer bytes from the input part of the buffer to the output
-  part.
+  This function returns all incoming data found in the channel buffer and remove
+  them from the buffer.
+
+  If there is no incoming data and the channel can't receive more data, a 'nil'
+  value is returned.
 
   :param class_channel channel: The manipulated Channel.
-  :param integer int: The amount of data which will be forwarded.
+  :returns: a string containing all the data found or nil.
 
-.. js:function:: Channel.is_full(channel)
+  .. warning::
+     This function is deprecated. :js:func:`Channel.data()` must be used to
+     retrieve data followed by a call to :js:func:`Channel:remove()` to remove
+     data.
 
-  This function returns true if the buffer channel is full.
+     .. code-block:: lua
 
-  :returns: a boolean
+       local data = chn:data()
+       chn:remove(0, data:len())
+
+     ..
+
+.. js:function:: Channel.getline(channel)
+
+  **DEPRECATED**
+
+  This function returns the first line found in incoming data of the channel
+  buffer, including the '\n'. The returned data are removed from the buffer. If
+  no line is found, this function yields to wait for more data, except if the
+  channel can't receive more data. In this case all data are returned.
+
+  If there is no incoming data and the channel can't receive more data, a 'nil'
+  value is returned.
+
+  :param class_channel channel: The manipulated Channel.
+  :returns: a string containing the line found or nil.
+
+  .. warning::
+     This function is depdrecated. :js:func:`Channel.line()` must be used to
+     retrieve a line followed by a call to :js:func:`Channel:remove()` to remove
+     data.
+
+     .. code-block:: lua
+
+       local line = chn:line(0, -1)
+       chn:remove(0, line:len())
+
+     ..
+
+.. js:function:: Channel.get_in_len(channel)
+
+  **DEPDRECATED**
+
+  This function returns the length of the input part of the buffer.
+
+  :param class_channel channel: The manipulated Channel.
+  :returns: an integer containing the amount of available bytes.
+
+  .. warning::
+     This function is deprecated. :js:func:`Channel.input()` must be used
+     instead.
+
+.. js:function:: Channel.get_out_len(channel)
+
+  **DEPDRECATED**
+
+  This function returns the length of the output part of the buffer.
+
+  :param class_channel channel: The manipulated Channel.
+  :returns: an integer containing the amount of available bytes.
+
+  .. warning::
+     This function is deprecated. :js:func:`Channel.output()` must be used
+     instead.
 
 .. _http_class:
 
diff --git a/src/hlua.c b/src/hlua.c
index 1e95ac9..5c2d3cc 100644
--- a/src/hlua.c
+++ b/src/hlua.c
@@ -2931,23 +2931,15 @@
 
 /* Copies <len> bytes of data present in the channel's buffer, starting at the
 * offset <offset>, and put it in a LUA string variable. It is the caller
-* responsibility to ensure <len> and <offset> are valid. If some data are copied
-* (len != 0), it returns the length of the built string. If no data are copied
-* (len == 0), it returns -1 and push a nil value in the stack if the channel's
-* input is closed. Otherwise it returns 0.
+* responsibility to ensure <len> and <offset> are valid. It always return the
+* length of the built string. <len> may be 0, in this case, an empty string is
+* created and 0 is returned.
 */
 static inline int _hlua_channel_dup(struct channel *chn, lua_State *L, size_t offset, size_t len)
 {
 	size_t block1, block2;
 	luaL_Buffer b;
 
-	if (unlikely(len == 0)) {
-		if (channel_input_closed(chn) || HLUA_CANT_YIELD(hlua_gethlua(L))) {
-			lua_pushnil(L);
-                        return -1;
-		}
-	}
-
 	block1 = len;
 	if (block1 > b_contig_data(&chn->buf, b_peek_ofs(&chn->buf, offset)))
 		block1 = b_contig_data(&chn->buf, b_peek_ofs(&chn->buf, offset));
@@ -2961,30 +2953,6 @@
 	return len;
 }
 
-/* Copies the first line (including the LF) in the channel's buffer, starting at
- * the offset <offset>, and put it in a LUA string variable. It copies at most
- * <len> bytes. It is the caller responsibility to ensure <len> and <offset> are
- * valid. If LF is found, the line is copied. If no LF is found and the no more
- * data can be received (channel's input is closed or full), <len> bytes are
- * copied. In both cases, the length of the built string is returned. Otherwise
- * nothing is copied, waiting for more data and 0 is returned.
- */
-static inline int _hlua_channel_dupline(struct channel *chn, lua_State *L, size_t offset, size_t len)
-{
-	size_t l;
-
-	for (l = 0; l < len; l++) {
-		if (*(b_peek(&chn->buf, offset+l)) == '\n')
-			return _hlua_channel_dup(chn, L, offset, l+1);
-	}
-
-	/* No LF found, and the channel may still receive new data, so wait */
-	if (!HLUA_CANT_YIELD(chn_strm(chn)->hlua) && !channel_input_closed(chn) && channel_may_recv(chn))
-		return 0;
-
-	return _hlua_channel_dup(chn, L, offset, l);
-}
-
 /* Inserts the string <str> to the channel's buffer at the offset <offset>. This
  * function returns -1 if data cannot be copied. Otherwise, it returns the
  * number of bytes copied.
@@ -2997,7 +2965,7 @@
 	if (unlikely(istlen(str) == 0))
 		goto end;
 
-	if (istlen(str) > c_room(chn) || channel_input_closed(chn)) {
+	if (istlen(str) > c_room(chn)) {
 		ret = -1;
 		goto end;
 	}
@@ -3019,65 +2987,161 @@
 	b_sub(&chn->buf, len);
 }
 
-/* "_hlua_channel_dup" wrapper. If no data are available, it returns
- * a yield. This function keep the data in the buffer.
+/* Copies input data in the channel's buffer. It is possible to set a specific
+ * offset (0 by default) and a length (all remaining input data starting for the
+ * offset by default). If there is not enough input data and more data can be
+ * received, this function yields.
  */
-__LJMP static int hlua_channel_dup_yield(lua_State *L, int status, lua_KContext ctx)
+__LJMP static int hlua_channel_get_data_yield(lua_State *L, int status, lua_KContext ctx)
 {
 	struct channel *chn;
 	int offset = 0, len = 0;
 
 	chn = MAY_LJMP(hlua_checkchannel(L, 1));
-	offset = co_data(chn);
-	len = ci_data(chn);
-	if (_hlua_channel_dup(chn, L, offset, len) == 0)
-		MAY_LJMP(hlua_yieldk(L, 0, 0, hlua_channel_dup_yield, TICK_ETERNITY, 0));
+
+	if (!ci_data(chn) && channel_input_closed(chn)) {
+		lua_pushnil(L);
+		return 1;
+	}
+
+	if (lua_gettop(L) > 1) {
+		offset = MAY_LJMP(luaL_checkinteger(L, 2));
+		if (offset < 0)
+			offset = MAX(0, ci_data(chn) + offset);
+		if (offset > ci_data(chn)) {
+			lua_pushfstring(L, "offset out of range.");
+			WILL_LJMP(lua_error(L));
+		}
+	}
+	len = ci_data(chn) - offset;
+	if (lua_gettop(L) == 3) {
+		len = MAY_LJMP(luaL_checkinteger(L, 3));
+		if (!len)
+			goto dup;
+		if (len == -1)
+			len = global.tune.bufsize;
+		if (len < 0) {
+			lua_pushfstring(L, "length out of range.");
+			WILL_LJMP(lua_error(L));
+		}
+	}
+
+	if (offset + len > ci_data(chn)) {
+		if (!HLUA_CANT_YIELD(hlua_gethlua(L)) && !channel_input_closed(chn) && channel_may_recv(chn)) {
+			/* Yield waiting for more data, as requested */
+			MAY_LJMP(hlua_yieldk(L, 0, 0, hlua_channel_get_data_yield, TICK_ETERNITY, 0));
+		}
+		len = ci_data(chn) - offset;
+	}
+
+  dup:
+	_hlua_channel_dup(chn, L, co_data(chn) + offset, len);
 	return 1;
 }
 
-/* Check arguments for the function "hlua_channel_dup_yield". */
-__LJMP static int hlua_channel_dup(lua_State *L)
+/* Copies the first line (including the trailing LF) of input data in the
+ * channel's buffer. It is possible to set a specific offset (0 by default) and
+ * a length (all remaining input data starting for the offset by default). If
+ * there is not enough input data and more data can be received, the function
+ * yields. If a length is explicitly specified, no more data are
+ * copied. Otherwise, if no LF is found and more data can be received, this
+ * function yields.
+ */
+__LJMP static int hlua_channel_get_line_yield(lua_State *L, int status, lua_KContext ctx)
 {
 	struct channel *chn;
+	int l, offset = 0, len = 0;
 
-	MAY_LJMP(check_args(L, 1, "dup"));
 	chn = MAY_LJMP(hlua_checkchannel(L, 1));
-	if (IS_HTX_STRM(chn_strm(chn))) {
-		lua_pushfstring(L, "Cannot manipulate HAProxy channels in HTTP mode.");
-		WILL_LJMP(lua_error(L));
+
+	if (!ci_data(chn) && channel_input_closed(chn)) {
+		lua_pushnil(L);
+		return 1;
+	}
+
+	if (lua_gettop(L) > 1) {
+		offset = MAY_LJMP(luaL_checkinteger(L, 2));
+		if (offset < 0)
+			offset = MAX(0, ci_data(chn) + offset);
+		if (offset > ci_data(chn)) {
+			lua_pushfstring(L, "offset out of range.");
+			WILL_LJMP(lua_error(L));
+		}
 	}
-	return MAY_LJMP(hlua_channel_dup_yield(L, 0, 0));
+	len = ci_data(chn) - offset;
+	if (lua_gettop(L) == 3) {
+		len = MAY_LJMP(luaL_checkinteger(L, 3));
+		if (!len)
+			goto dup;
+		if (len == -1)
+			len = global.tune.bufsize;
+		if (len < 0) {
+			lua_pushfstring(L, "length out of range.");
+			WILL_LJMP(lua_error(L));
+		}
+	}
+
+	for (l = 0; l < len; l++) {
+		if (l + offset >= ci_data(chn))
+			break;
+		if (*(b_peek(&chn->buf, co_data(chn) + offset + l)) == '\n') {
+			len = l+1;
+			goto dup;
+		}
+	}
+
+	if (offset + len > ci_data(chn)) {
+		if (!HLUA_CANT_YIELD(hlua_gethlua(L)) && !channel_input_closed(chn) && channel_may_recv(chn)) {
+			/* Yield waiting for more data */
+			MAY_LJMP(hlua_yieldk(L, 0, 0, hlua_channel_get_line_yield, TICK_ETERNITY, 0));
+		}
+		len = ci_data(chn) - offset;
+	}
+
+  dup:
+	_hlua_channel_dup(chn, L, co_data(chn) + offset, len);
+	return 1;
 }
 
-/* "_hlua_channel_dup" + "_hlua_channel_erase" wrapper. If no data are
- * available, it returns a yield. This function consumes the data in the
- * buffer. It returns a string containing the data or a nil pointer if no data
- * are available and the channel is closed.
+/* [ DEPRECATED ]
+ *
+ * Duplicate all input data foud in the channel's buffer. The data are not
+ * removed from the buffer. This function relies on _hlua_channel_dup().
  */
-__LJMP static int hlua_channel_get_yield(lua_State *L, int status, lua_KContext ctx)
+__LJMP static int hlua_channel_dup(lua_State *L)
 {
 	struct channel *chn;
 	int offset = 0, len = 0;
-	int ret;
 
+	MAY_LJMP(check_args(L, 1, "dup"));
 	chn = MAY_LJMP(hlua_checkchannel(L, 1));
+	if (IS_HTX_STRM(chn_strm(chn))) {
+		lua_pushfstring(L, "Cannot manipulate HAProxy channels in HTTP mode.");
+		WILL_LJMP(lua_error(L));
+	}
 	offset = co_data(chn);
 	len = ci_data(chn);
 
-	ret = _hlua_channel_dup(chn, L, offset, len);
-	if (unlikely(ret == 0))
-		MAY_LJMP(hlua_yieldk(L, 0, 0, hlua_channel_get_yield, TICK_ETERNITY, 0));
-	if (unlikely(ret == -1))
+	if (!ci_data(chn) && channel_input_closed(chn)) {
+		lua_pushnil(L);
 		return 1;
+	}
 
-	_hlua_channel_delete(chn, offset, ret);
+	_hlua_channel_dup(chn, L, offset, len);
 	return 1;
 }
 
-/* Check arguments for the function "hlua_channel_get_yield". */
+/* [ DEPRECATED ]
+ *
+ * Get all input data foud in the channel's buffer. The data are removed from
+ * the buffer after the copy. This function relies on _hlua_channel_dup() and
+ * _hlua_channel_delete().
+ */
 __LJMP static int hlua_channel_get(lua_State *L)
 {
 	struct channel *chn;
+	int offset = 0, len = 0;
+	int ret;
 
 	MAY_LJMP(check_args(L, 1, "get"));
 	chn = MAY_LJMP(hlua_checkchannel(L, 1));
@@ -3085,7 +3149,17 @@
 		lua_pushfstring(L, "Cannot manipulate HAProxy channels in HTTP mode.");
 		WILL_LJMP(lua_error(L));
 	}
+	offset = co_data(chn);
+	len = ci_data(chn);
+
+	if (!ci_data(chn) && channel_input_closed(chn)) {
+		lua_pushnil(L);
+		return 1;
+	}
-	return MAY_LJMP(hlua_channel_get_yield(L, 0, 0));
+
+	ret = _hlua_channel_dup(chn, L, offset, len);
+	_hlua_channel_delete(chn, offset, ret);
+	return 1;
 }
 
 /* This functions consumes and returns one line. If the channel is closed,
@@ -3096,6 +3170,7 @@
 __LJMP static int hlua_channel_getline_yield(lua_State *L, int status, lua_KContext ctx)
 {
 	struct channel *chn;
+	size_t l;
 	int offset = 0, len = 0;
 	int ret;
 
@@ -3103,17 +3178,33 @@
 	offset = co_data(chn);
 	len = ci_data(chn);
 
-	ret = _hlua_channel_dupline(chn, L, offset, len);
-	if (ret == 0)
-		MAY_LJMP(hlua_yieldk(L, 0, 0, hlua_channel_getline_yield, TICK_ETERNITY, 0));
-	if (unlikely(ret == -1))
+	if (!ci_data(chn) && channel_input_closed(chn)) {
+		lua_pushnil(L);
 		return 1;
+	}
+
+	for (l = 0; l < len; l++) {
+		if (*(b_peek(&chn->buf, offset+l)) == '\n') {
+			len = l+1;
+			goto dup;
+		}
+	}
+
+	if (!HLUA_CANT_YIELD(hlua_gethlua(L)) && !channel_input_closed(chn) && channel_may_recv(chn)) {
+		/* Yield waiting for more data */
+		MAY_LJMP(hlua_yieldk(L, 0, 0, hlua_channel_getline_yield, TICK_ETERNITY, 0));
+	}
 
+  dup:
+	ret = _hlua_channel_dup(chn, L, co_data(chn) + offset, len);
 	_hlua_channel_delete(chn, offset, ret);
 	return 1;
 }
 
-/* Check arguments for the function "hlua_channel_getline_yield". */
+/* [ DEPRECATED ]
+ *
+ * Check arguments for the function "hlua_channel_getline_yield".
+ */
 __LJMP static int hlua_channel_getline(lua_State *L)
 {
 	struct channel *chn;
@@ -3127,10 +3218,48 @@
 	return MAY_LJMP(hlua_channel_getline_yield(L, 0, 0));
 }
 
-/* This function takes a string as argument, and append it at the input side of
- * channel. If data cannot be copied, because there is not enough space to do so
- * or because the channel is closed, it returns -1. Otherwise, it returns the
- * amount of data written (the string length). This function does not yield.
+/* Retrieves a given amount of input data at the given offset. By default all
+ * available input data are returned. The offset may be negactive to start from
+ * the end of input data. The length may be -1 to set it to the maximum buffer
+ * size.
+ */
+__LJMP static int hlua_channel_get_data(lua_State *L)
+{
+	struct channel *chn;
+
+	if (lua_gettop(L) < 1 || lua_gettop(L) > 3)
+		WILL_LJMP(luaL_error(L, "'data' expects at most 2 arguments"));
+	chn = MAY_LJMP(hlua_checkchannel(L, 1));
+	if (IS_HTX_STRM(chn_strm(chn))) {
+		lua_pushfstring(L, "Cannot manipulate HAProxy channels in HTTP mode.");
+		WILL_LJMP(lua_error(L));
+	}
+	return MAY_LJMP(hlua_channel_get_data_yield(L, 0, 0));
+}
+
+/* Retrieves a given amount of input data at the given offset. By default all
+ * available input data are returned. The offset may be negactive to start from
+ * the end of input data. The length may be -1 to set it to the maximum buffer
+ * size.
+ */
+__LJMP static int hlua_channel_get_line(lua_State *L)
+{
+	struct channel *chn;
+
+	if (lua_gettop(L) < 1 || lua_gettop(L) > 3)
+		WILL_LJMP(luaL_error(L, "'line' expects at most 2 arguments"));
+	chn = MAY_LJMP(hlua_checkchannel(L, 1));
+	if (IS_HTX_STRM(chn_strm(chn))) {
+		lua_pushfstring(L, "Cannot manipulate HAProxy channels in HTTP mode.");
+		WILL_LJMP(lua_error(L));
+	}
+	return MAY_LJMP(hlua_channel_get_line_yield(L, 0, 0));
+}
+
+/* Appends a string into the input side of channel. It returns the length of the
+ * written string, or -1 if the channel is closed or if the buffer size is too
+ * little for the data. 0 may be returned if nothing is copied. This function
+ * does not yield.
  */
 __LJMP static int hlua_channel_append(lua_State *L)
 {
@@ -3142,7 +3271,6 @@
 	MAY_LJMP(check_args(L, 2, "append"));
 	chn = MAY_LJMP(hlua_checkchannel(L, 1));
 	str = MAY_LJMP(luaL_checklstring(L, 2, &len));
-
 	if (IS_HTX_STRM(chn_strm(chn))) {
 		lua_pushfstring(L, "Cannot manipulate HAProxy channels in HTTP mode.");
 		WILL_LJMP(lua_error(L));
@@ -3153,35 +3281,146 @@
 	return 1;
 }
 
-/* The function replaces input data from the channel by the string passed as
- * argument. Before clearing the buffer, we take care the copy will be
- * possible. It returns the amount of data written (the string length) on
- * success or -1 if the copy is not performed. This function does not yield.
+/* Prepends a string into the input side of channel. It returns the length of the
+ * written string, or -1 if the channel is closed or if the buffer size is too
+ * little for the data. 0 may be returned if nothing is copied. This function
+ * does not yield.
  */
-__LJMP static int hlua_channel_set(lua_State *L)
+__LJMP static int hlua_channel_prepend(lua_State *L)
 {
 	struct channel *chn;
 	const char *str;
-	size_t len;
+	size_t sz;
 	int ret;
 
-	MAY_LJMP(check_args(L, 2, "set"));
+	MAY_LJMP(check_args(L, 2, "prepend"));
 	chn = MAY_LJMP(hlua_checkchannel(L, 1));
-	str = MAY_LJMP(luaL_checklstring(L, 2, &len));
+	str = MAY_LJMP(luaL_checklstring(L, 2, &sz));
+	if (IS_HTX_STRM(chn_strm(chn))) {
+		lua_pushfstring(L, "Cannot manipulate HAProxy channels in HTTP mode.");
+		WILL_LJMP(lua_error(L));
+	}
+
+	ret = _hlua_channel_insert(chn, L, ist2(str, sz), co_data(chn));
+	lua_pushinteger(L, ret);
+	return 1;
+}
+
+/* Inserts a given amount of input data at the given offset by a string
+ * content. By default the string is appended at the end of input data. It
+ * returns the length of the written string, or -1 if the channel is closed or
+ * if the buffer size is too little for the data.
+ */
+__LJMP static int hlua_channel_insert_data(lua_State *L)
+{
+	struct channel *chn;
+	const char *str;
+	size_t sz;
+	int ret, offset;
+
+	if (lua_gettop(L) < 2 || lua_gettop(L) > 3)
+		WILL_LJMP(luaL_error(L, "'insert' expects at least 1 argument and at most 2 arguments"));
+	chn = MAY_LJMP(hlua_checkchannel(L, 1));
+	str = MAY_LJMP(luaL_checklstring(L, 2, &sz));
+	offset = ci_data(chn);
+	if (lua_gettop(L) > 2)
+		offset = MAY_LJMP(luaL_checkinteger(L, 3));
+	if (IS_HTX_STRM(chn_strm(chn))) {
+		lua_pushfstring(L, "Cannot manipulate HAProxy channels in HTTP mode.");
+		WILL_LJMP(lua_error(L));
+	}
+
+	if (offset < 0)
+		offset = MAX(0, ci_data(chn) + offset);
+
+	if (offset > ci_data(chn)) {
+		lua_pushfstring(L, "offset out of range.");
+		WILL_LJMP(lua_error(L));
+	}
+
+	ret = _hlua_channel_insert(chn, L, ist2(str, sz), co_data(chn) + offset);
+	lua_pushinteger(L, ret);
+	return 1;
+}
+/* Replaces a given amount of input data at the given offset by a string
+ * content. By default all remaining data are removed (offset = 0 and len =
+ * -1). It returns the length of the written string, or -1 if the channel is
+ * closed or if the buffer size is too little for the data.
+ */
+__LJMP static int hlua_channel_set_data(lua_State *L)
+{
+	struct channel *chn;
+	const char *str;
+	size_t sz;
+	int ret, offset = 0, len = -1;
 
+	if (lua_gettop(L) < 2 || lua_gettop(L) > 4)
+		WILL_LJMP(luaL_error(L, "'set' expects at least 1 argument and at most 3 arguments"));
+	chn = MAY_LJMP(hlua_checkchannel(L, 1));
+	str = MAY_LJMP(luaL_checklstring(L, 2, &sz));
+	if (lua_gettop(L) > 2)
+		offset = MAY_LJMP(luaL_checkinteger(L, 3));
+	if (lua_gettop(L) == 4)
+		len = MAY_LJMP(luaL_checkinteger(L, 4));
 	if (IS_HTX_STRM(chn_strm(chn))) {
 		lua_pushfstring(L, "Cannot manipulate HAProxy channels in HTTP mode.");
 		WILL_LJMP(lua_error(L));
 	}
 
+	if (offset < 0)
+		offset = MAX(0, ci_data(chn) + offset);
+	if (len < 0)
+		len = ci_data(chn) - offset;
+
+	if (offset + len > ci_data(chn)) {
+		lua_pushfstring(L, "offset or length out of range.");
+		WILL_LJMP(lua_error(L));
+	}
+
 	/* Be sure we can copied the string once input data will be removed. */
-	if (len > c_room(chn) + ci_data(chn) || channel_input_closed(chn))
+	if (sz > c_room(chn) + len)
 		lua_pushinteger(L, -1);
 	else {
-		_hlua_channel_delete(chn, co_data(chn), ci_data(chn));
-		ret = _hlua_channel_insert(chn, L, ist2(str, len), co_data(chn));
+		_hlua_channel_delete(chn, co_data(chn) + offset, len);
+		ret = _hlua_channel_insert(chn, L, ist2(str, sz), co_data(chn) + offset);
 		lua_pushinteger(L, ret);
 	}
+	return 1;
+}
+
+/* Removes a given amount of input data at the given offset. By default all
+ * intput data are removed (offset = 0 and len = -1). It returns the amount of
+ * the removed data.
+ */
+__LJMP static int hlua_channel_del_data(lua_State *L)
+{
+	struct channel *chn;
+	int offset = 0, len = -1;
+
+	if (lua_gettop(L) < 1 || lua_gettop(L) > 3)
+		WILL_LJMP(luaL_error(L, "'remove' expects at most 2 arguments"));
+	chn = MAY_LJMP(hlua_checkchannel(L, 1));
+	if (lua_gettop(L) > 1)
+		offset = MAY_LJMP(luaL_checkinteger(L, 2));
+	if (lua_gettop(L) == 3)
+		len = MAY_LJMP(luaL_checkinteger(L, 3));
+	if (IS_HTX_STRM(chn_strm(chn))) {
+		lua_pushfstring(L, "Cannot manipulate HAProxy channels in HTTP mode.");
+		WILL_LJMP(lua_error(L));
+	}
+
+	if (offset < 0)
+		offset = MAX(0, ci_data(chn) + offset);
+	if (len < 0)
+		len = ci_data(chn) - offset;
+
+	if (offset + len > ci_data(chn)) {
+		lua_pushfstring(L, "offset or length out of range.");
+		WILL_LJMP(lua_error(L));
+	}
+
+	_hlua_channel_delete(chn, co_data(chn) + offset, len);
+	lua_pushinteger(L, len);
 	return 1;
 }
 
@@ -3362,7 +3601,7 @@
 {
 	struct channel *chn;
 
-	MAY_LJMP(check_args(L, 1, "get_in_len"));
+	MAY_LJMP(check_args(L, 1, "input"));
 	chn = MAY_LJMP(hlua_checkchannel(L, 1));
 	if (IS_HTX_STRM(chn_strm(chn))) {
 		struct htx *htx = htxbuf(&chn->buf);
@@ -3387,6 +3626,17 @@
 	return 1;
 }
 
+/* Returns true if the channel may still receive data. */
+__LJMP static int hlua_channel_may_recv(lua_State *L)
+{
+	struct channel *chn;
+
+	MAY_LJMP(check_args(L, 1, "may_recv"));
+	chn = MAY_LJMP(hlua_checkchannel(L, 1));
+	lua_pushboolean(L, (!channel_input_closed(chn) && channel_may_recv(chn)));
+	return 1;
+}
+
 /* Returns true if the channel is the response channel. */
 __LJMP static int hlua_channel_is_resp(lua_State *L)
 {
@@ -3406,7 +3656,7 @@
 {
 	struct channel *chn;
 
-	MAY_LJMP(check_args(L, 1, "get_out_len"));
+	MAY_LJMP(check_args(L, 1, "output"));
 	chn = MAY_LJMP(hlua_checkchannel(L, 1));
 	lua_pushinteger(L, co_data(chn));
 	return 1;
@@ -8912,18 +9162,28 @@
 	lua_newtable(L);
 
 	/* Register . */
-	hlua_class_function(L, "get",         hlua_channel_get);
-	hlua_class_function(L, "dup",         hlua_channel_dup);
-	hlua_class_function(L, "getline",     hlua_channel_getline);
-	hlua_class_function(L, "set",         hlua_channel_set);
+	hlua_class_function(L, "data",        hlua_channel_get_data);
+	hlua_class_function(L, "line",        hlua_channel_get_line);
+	hlua_class_function(L, "set",         hlua_channel_set_data);
+	hlua_class_function(L, "remove",      hlua_channel_del_data);
 	hlua_class_function(L, "append",      hlua_channel_append);
+	hlua_class_function(L, "prepend",     hlua_channel_prepend);
+	hlua_class_function(L, "insert",      hlua_channel_insert_data);
 	hlua_class_function(L, "send",        hlua_channel_send);
 	hlua_class_function(L, "forward",     hlua_channel_forward);
-	hlua_class_function(L, "get_in_len",  hlua_channel_get_in_len);
-	hlua_class_function(L, "get_out_len", hlua_channel_get_out_len);
+	hlua_class_function(L, "input",       hlua_channel_get_in_len);
+	hlua_class_function(L, "output",      hlua_channel_get_out_len);
+	hlua_class_function(L, "may_recv",    hlua_channel_may_recv);
 	hlua_class_function(L, "is_full",     hlua_channel_is_full);
 	hlua_class_function(L, "is_resp",     hlua_channel_is_resp);
 
+	/* Deprecated API */
+	hlua_class_function(L, "get",         hlua_channel_get);
+	hlua_class_function(L, "dup",         hlua_channel_dup);
+	hlua_class_function(L, "getline",     hlua_channel_getline);
+	hlua_class_function(L, "get_in_len",  hlua_channel_get_in_len);
+	hlua_class_function(L, "get_out_len", hlua_channel_get_out_len);
+
 	lua_rawset(L, -3);
 
 	/* Register previous table in the registry with reference and named entry. */