netcgi1 programs to netcgi2
The library netcgi2 is a revised version of the old cgi library
which is now also referred to as netcgi1.
As any software, netcgi1 aged, and suffered more and more from
inadequate interfaces. Because of this it became necessary to
improve the interfaces from grounds up. The result is netcgi2,
a new major version that tries to continue the good parts of
netcgi1 while replacing its problematic edges.
When this text is written, netcgi2 is still being developed, and
subject of discussion.
It is not possible to use netcgi1 and netcgi2 at the same time in the
same application. This means that one cannot gradually upgrade from
netcgi1 to netcgi2 by using more and more of the netcgi2 features.
Instead of this, it is necessary to switch from netcgi1 to netcgi2
at one point in the lifetime of the web application.
The main benefit is that you have access to the newest netcgi2
features. There are already a number of connectors that are not
present in netcgi1 (newer AJP protocol version, SCGI). Furthermore,
new features will only be added to netcgi2.
However, if your application is already at or near its end of lifetime,
there is no need to port it to netcgi2. The netcgi1 library will
remain in Ocamlnet 2, and bugs will be fixed.
The new organization is very simple:
Netcgi defines all basic types. Previously, this was done in the
  two modules Netcgi_env and Netcgi_typesNetcgi_c implementing
  it. Especially the classic CGI connector is now in Netcgi_cgi.
  Previously, the CGI connector was defined in Netcgi, and there
  used to be several modules per connector.Netcgi_common defines service functions to define new connectors.Netcgi1_compat trying to ease porting. See
below for a discussion.
Most of the types remain the same, or almost the same. A few changes have been done:
Netcgi.cgi_argument is no longer writable.
  Furthermore, the list of arguments in a Netcgi.cgi_activation can no longer
  be modified. There are some new service functions to modify lists
  of arguments in case one needs such a list.Netcgi.Cookie.Netcgi_common.HTTP exception can be used to exit
  from a processor at any time. There is the notion of an exception
  handler for web-related exceptions.Netcgi.cgi_environment have been simplified.
  It is only distinguished between two states: Output headers have been/
  have not been sent. Other processing states are hidden by the
  implementation.Netcgi.cgi_activation.at_exit
  method.
In the long term this is the best strategy. In principle, one has to distinguish between
netcgi values, andnetcgi application with the web
  server.netcgi values do not change much. For example, the function
web_page for netcgi1
 
(* This is [netcgi1] code! *)
let web_page (cgi : Netcgi_types.cgi_activation) =
  let webarg = cgi # argument_value "webarg" in
  cgi # set_header();
  cgi # output # output_string ("The argument is: " ^ webarg)
would read in the version for netcgi2 as follows:
 
(* This is [netcgi2] code! *)
let web_page (cgi : Netcgi.cgi_activation) =
  let webarg = cgi # argument_value "webarg" in
  cgi # set_header();
  cgi # output # output_string ("The argument is: " ^ webarg)
The only change is that the type cgi_activation is now defined
in the module Netcgi and no longer in Netcgi_types. It is expected
that this simple way of porting applies to almost all parts of
netcgi applications.
By the way, the type cgi_activation can now be abbreviated as cgi,
as this is the type name that needs to be written down most
frequently.
In netcgi1, the CGI connector is selected by instantiating the class
Netcgi.std_activation, as in:
(* This is [netcgi1] code! *)
let cgi = new Netcgi.std_activation() in 
process cgi
It is assumed that process is a function taking a cgi_activation
as argument, and processing the request.
The corresponding netcgi2 call is:
(* This is [netcgi2] code! *)
Netcgi_cgi.run process
As you see, Netcgi_cgi.run is now responsible for calling process.
In netcgi1 there are several ways of using FastCGI. The most common is
to call Netcgi_fcgi.serv as in:
(* This is [netcgi1] code! *)
Netcgi_fcgi.serv process optype
It is assumed that process is a function taking a cgi_activation
as argument, and processing the request. optype is a valid
operating type.
The corresponding netcgi2 call is:
(* This is [netcgi2] code! *)
let process' cgi = process (cgi :> Netcgi.cgi_activation) in
Netcgi_fcgi.run ~output_type:optype process'
Note that the argument of process' is a slightly extended version
of cgi_activation, so you usually need the coercion to cut off the
additional part of the object interface.
The new connector supports now the AJP version 1.3 - this is the default
version used by Jakarta and mod_jk. In netcgi1, only version 1.2 of
the AJP protocol was supported. The new protocol version is no big
improvement, however. It uses a slightly more compact representation
of the data. The biggest plus is better support of SSL.
In netcgi1 there was some special machinery around the AJP connector
to create worker processes. This code has been completely removed
in favor of netplex, the new general-purpose server framework.
Because of that, porting AJP applications is probably a bit of work,
and we cannot give a receipt here how to do that.
Netcgi1_compat
If you want to use the new connectors but currently do not have time
to check all your code for changes, there is a special helper module
called Netcgi1_compat that provides a netcgi1-compatible API
on top of either netcgi1 or netcgi2.
Because Netcgi1_compat is available in both netcgi1 and netcgi2
you can write code that can be compiled for both versions without
needing to change anything. Note, however, that this module is not
100% identical in both versions - the netcgi2 version includes some
additional functions that converts values from their netcgi1
representation to their netcgi2 representation and vice versa.
Unfortunately, this makes the two versions of this module 
binary-incompatible, so you have to recompile your code for either
netcgi1 or netcgi2.
The Netcgi1_compat module simply contains the relevant parts of
the netcgi1 API as submodules. That means you can access
netcgi1 version of the module Netcgi_types as 
  Netcgi1_compat.Netcgi_typesnetcgi1 version of the module Netcgi_env as 
  Netcgi1_compat.Netcgi_envnetcgi1 version of the module Netcgi as Netcgi1_compat.Netcginetcgi1 are not covered by the compatibility API.
In Netcgi, the custom_activation class has been left out.
You can usually port code to using this API by either
Netcgi1., e.g.
  new Netcgi.std_activation() would be turned into
  new Netcgi1_compat.Netcgi.std_activation()Netcgi1 at the beginning of each .ml and .mli
  file by an open Netcgi1_compat.Netcgi.std_activation, the netcgi1 way of creating a
connector for classic CGI, there are no connectors in the 
compatibility API. If you need one, you must take it directly 
from either netcgi1 or netcgi2. For example, to connect using 
FastCGI:
(* This is code for both [netcgi1] and [netcgi2]! *)
let process (cgi : Netcgi1_compat.Netcgi.cgi_activation) =
  ...
(* This is [netcgi2] code! *)
let process_netcgi2 cgi2 =
  let cgi1 = Netcgi1_compat.Netcgi_types.to_compat_activation cgi2 in
  process cgi1
Netcgi_fcgi.run ~output_type process'