Please report new issues athttps://github.com/DOCGroup
IONA has made ZIOP available and they want to standardize this as part of the OMG spec (proposal at the Anaheim 2006 meeting). It is not a pluggable protocol, it is more a plugin mechanism to let applications but also the ORB use compression. Seems a good addition for TAO for people interested in bandwith. I have looked at ZIOP and some parts could be improved so that memory allocations are reduced. See also http://www.iona.it/support/docs/orbix/6.2/reference/corba_pref/cpp/IT_ZIOP.html
accept
IONA presented ZIOP at the Anaheim 2006 meeting. We are going to look if we can implement this and work on the next OMG spec step which hopefully can be presented at Brussel 2007
the orbix api as it is now #ifndef _ORBIX_PDK_ZIOP_IDL_ #define _ORBIX_PDK_ZIOP_IDL_ // // Copyright (c) 2004 IONA Technologies PLC. All Rights Reserved. // // WARNING! This Plug-In Development Kit allows you to access internal // portions of IONA's software. If you use this PDK to develop custom // plug-ins, IONA cannot ensure binary compatibility with your custom // plug-ins. The contents of this file may change without notice and your // custom plug-ins may need to be re-linked, re-compiled, re-factored, or // even re-written to work with a later version of IONA's software. By // using this PDK you assume this risk and acknowledge that IONA shall not // be responsible for ensuring binary compatibility or for the results of // your use. #include <omg/orb.idl> #include <orbix_pdk/buffer.idl> #include <orbix_pdk/ior.idl> #include <orbix/policy_base.idl> /** * The IT_ZIOP module defines interfaces, exceptions, types and values for the * IONA ZIOP Compression plug-in. * This plug-in provides optional compression of all types of GIOP messages via * a message level interceptor that is installed between the GIOP interceptor * and the transport interceptor (e.g., IIOP, IIOP_TLS, etc). * The idl defines the interfaces of the plug-in to register compression * algorithms, the ZIOP IOR Component, and the Policies associated with the * plug-in. */ module IT_ZIOP { /** * Exception thrown when an error occurs during a compress or decompress * operation. */ exception CompressionException { string reason; }; /** * Exception thrown if a CompressorFactory with the same CompressorId is * already registered with the CompressionManager. */ exception FactoryAlreadyRegistered { }; /** * Exception thrown if a CompressorId is not known. */ exception UnknownCompressorId { }; /** * CompressorId type. */ typedef unsigned long CompressorId; local interface CompressorFactory; /** * Compressor - abstraction of a GIOP message level compressor and * decompressor. */ local interface Compressor { /** * The CompressorFactory associated with this Compressor. */ readonly attribute CompressorFactory compressor_factory; /** * The (implementation and algorithm specific) compression level * associated with this Compressor. */ readonly attribute long compression_level; /** * Operation that compresses data contained in the source Buffer into * the target Buffer. If an error occurs during the compression, it * throws CompressionException */ void compress( in IT_Buffer::Buffer source, in IT_Buffer::Buffer target ) raises (CompressionException); /** * Operation that decompresses data contained in the source Buffer into * the target Buffer. If an error occurs during the decompression, it * throws CompressionException */ void decompress( in IT_Buffer::Buffer source, in IT_Buffer::Buffer target ) raises (CompressionException); }; /** * CompressorFactory - factory for Compressor instances with a particular * (implementation and algorithm specific) compression level. */ local interface CompressorFactory { /** * The CompressorId associated with this CompressorFactory */ readonly attribute CompressorId compressor_id; /** * The total number of compressed bytes read and written by Compressors * that were created by this CompressorFactory * (i.e. the "target" side of Compressor::compress and * the "source" side of Compressor::decompress operations). */ readonly attribute unsigned long long compressed_bytes; /** * The total number of uncompressed bytes read and written by * Compressors that were created by this CompressorFactory * (i.e. the "source" side of Compressor::compress and * the "target" side of Compressor::decompress operations). */ readonly attribute unsigned long long uncompressed_bytes; /** * The average compression achieved by Compressors that were created by * this CompressorFactory, usually a value between 0 and >=1. * (i.e. compressed_bytes divided by uncompressed_bytes). */ readonly attribute double average_compression; /** * Create a Compressor instance with the given compression level. */ Compressor get_compressor(in long compression_level); /** * Add a sample of compressed and uncompressed bytes. */ void add_sample( in unsigned long long compressed_bytes, in unsigned long long uncompressed_bytes ); }; /** * Sequence of CompressorFactory */ typedef sequence<CompressorFactory> CompressorFactorySeq; /** * Per-ORB interface to register and unregister CompressorFactories. * Initial reference: "IT_CompressionManager" */ local interface CompressionManager { /** * Register a new CompressorFactory */ void register_factory( in CompressorFactory compressor_factory ) raises (FactoryAlreadyRegistered); /** * Unregister a CompressorFactory with the given CompressorId from the * CompressionManager */ void unregister_factory( in CompressorId compressor_id ) raises (UnknownCompressorId); /** * Retrieve a CompressorFactory with the given CompressorId from the * CompressionManager */ CompressorFactory get_factory( in CompressorId compressor_id ) raises (UnknownCompressorId); /** * Create a Compressor with the given compression_level from the * CompressorFactory with the given CompressorId */ Compressor get_compressor( in CompressorId compressor_id, in long compression_level ) raises (UnknownCompressorId); /** * List all registered CompressorFactories */ CompressorFactorySeq get_factories( ); }; // // IOR Component for ZIOP // /** * The ZIOP IOR Component Tag Id */ const IOP::ComponentId TAG_IONA_ZIOP_COMPONENT = 0x49545F46; /** * The ZIOP IOR Component. Has an CompressorId attribute indicating the * compression algorithm supported by the server side. */ local interface CompressionComponent : IT_IOR::Component { readonly attribute CompressorId compressor_id; }; /** * Factory for ZIOP IOR Components */ local interface CompressionComponentFactory : IT_IOR::ComponentFactory { CompressionComponent get_compression_component( in CompressorId compressor_id ); }; // // ORB Policies for ZIOP // /** * Tag Id for CompressionEnablingPolicy */ const CORBA::PolicyType COMPRESSION_ENABLING_POLICY_ID = IT_PolicyBase::IONA_POLICY_ID + 0x46; /** * Tag Id for CompressorIdPolicy */ const CORBA::PolicyType COMPRESSOR_ID_POLICY_ID = IT_PolicyBase::IONA_POLICY_ID + 0x47; /** * The ZIOP CompressionEnablingPolicy. Has an boolean attribute indicating * if compression is enabled or not. */ local interface CompressionEnablingPolicy : CORBA::Policy { readonly attribute boolean compression_enabled; }; /** * The ZIOP CompressorIdPolicy. Has an CompressorId attribute indicating * the compression algorithm to be used. */ local interface CompressorIdPolicy : CORBA::Policy { readonly attribute CompressorId compressor_id; }; }; #endif /*!_ORBIX_PDK_ZIOP_IDL_*/
Andre Spiegel will implement ZIOP for JacORB
The current ZIOP implementation in Orbix does not specifically address compression for replies, i.e. from server to client. Maybe that is handled implicitly: if the client starts using compression towards the server, then the server feels free to use compression in its own replies. But that won't work: The client might not use compression in its own requests, because they are too small, but still expect compression in the replies, which could be much larger. Currently, there seems no way for the server to know whether the client is willing/able to accept compressed replies or not. So here's a proposal: Each side must explicitly allow compression, only then may the other side send compressed messages. The server can allow compression by publishing a ZIOP component in the IOR, or by sending a compression-allowing service context piggy-backed to one of its own replies. The client can allow compression by sending such a service context, piggy-backed to one of its own requests. It is a requirement in my project that compression can be switched on and off dynamically, without restarting server or client. This would be easy with the above scheme, if the compression service context can either allow or disallow compression. The service context could simply contain a list of allowed compressor ids; sending an empty list would mean that compression is no longer allowed.
Some more discussion points: Hi, > > But with the string we can't do much automatically, > couldn't we better have > > an enum and then values for out of memory, not enough room > in buffer, etc. > > If the error conditions are sufficiently different, and if > programs will > usually want to react differently to each of these conditions, then I > think it makes more sense to have separate exception types > for them. On > the other hand, if programs will usually just abort, or try again, no > matter what the actual condition is, then I think using a single > exception type with a "reason" message is enough. Probably when compress/decompress fails will be a technical reason. Zlib for example gives two possible error return values, one out of memory, other that the buffer is too small. Converting these to a message in the ORB is something I don't want to do, what if we have a Dutch or German UI? > It's a pity that IDL does not have exception inheritance, because then > you could have the best of both worlds, by making all the individual > exceptions subclasses of CompressionException. That's what > you would do > in Java. > > I don't really like the idea of having a separate enum field to > distinguish the errors. This is partly because enums are mapped very > awkwardly in the Java mapping. But that's just my two cents. Ok, but having multiple exceptions would lead to a bigger footprint but I do see your point. Let us wait until next week when the others are back from holiday to see how they respond. Btw, I have almost the Compression part ready for TAO. I only need to write a small regression test and then I will commit this to the main repository for TAO. Johnny
Hi, The type is ok, the minor codes are coming from an appendix of the spec. For the time being I have taken 44. Johnny > -----Original Message----- > From: Andre Spiegel [mailto:spiegel@gnu.org] > Sent: Wednesday, December 27, 2006 3:01 PM > To: Johnny Willemsen > Subject: RE: ZIOP Concepts > > > > A real minor detail, but what we should document/implement > is what happens > > if we call Compression_Manager::register_factory with a nil object > > reference, which system exception and then which minor code? > > I'd say BAD_PARAM. I have no idea how minor codes are > assigned. I have > never been able to find a comprehensive list of them. > > Andre > >
TAO 1.5.5 will have a compression library with compressormanager/factory and a zlib compressor. A small test is there, just testing if things work and some error situations. I have raised some questions to IONA about some interfaces and some details, Andre has raised some questions about how ZIOP itself should work. With this compression library available the next step is to add ZIOP to the ORB itself. For TAO this will result in an adapter and then ZIOP in a seperate library.
mine
to pool
added depends
still valid enhancement, taking this over
From IONA I'm afraid there's very little "official" documentation on the ZIOP implementation in Orbix, beyond the section in the chapter in the administrator and programmer's guides [1] [2] [3]. Here's a quick summary how ZIOP works: (1) The server indicates ZIOP support by adding a ZIOP component to it's IORs. This IRO Component contains a long(?) value that specifies the compressor ID (i.e. the compression algorithm) to be used by clients that also support ZIOP. (2) A ZIOP aware client would read the IOR component and if it supports the given compression algorithm it can then send compressed messages to the server and receive compressed replies. (3) The ZIOP protocol used for compressed messages uses two unused bit's in the GIOP message header to indicate that (a) compression is supported by the sending side and (b) if the message is compressed or not. If the message is compressed, the entire message body is compressed, i.e. anything after the initial 12 bytes. This allows the sending side not to compress a message (e.g. in Orbix it is possible to disable compression for messages that are smaller than a configures threshold) but still indicate the receiving side that compression is supported and that a reply may be sent compressed. This is the basic idea behind ZIOP in Orbix. Does this make sense? Is this along the lines you've been working on compression in TAO? [1] Orbix 6.3 C++ Programmer's Guide: http://www.iona.com/support/docs/orbix/6.3/develop/corba_pguide/cpp/pguide_cpp.pdf [2] Orbix 6.3 Java Programmer's Guide: http://www.iona.com/support/docs/orbix/6.3/develop/corba_pguide/java/pguide_java.pdf [3] Orbix 6.3 Administrator's Guide: http://www.iona.com/support/docs/orbix/6.3/admin/admin_guide/admin.pdf
I have created branch ZIOP2, merged all changes to that branch and I am working on getting this running using the compression library. First just the quick way to find the best places in the orb to place the adapters in, when things work after a first iteration I will then refactor things, make it cleaner, etc.
branch ZIOP2 has a draft solution that now works. Always compression enabled because no policies yet, but we do transfer compressed data, so debug log below, the string is 195 bytes long, the request send from client to server is in total 136. TAO (3660|416) - GIOP_Message_Base::dump_msg, recv GIOP v1.2 msg, 136 data bytes, my endian, Type Request[1] GIOP message - HEXDUMP 148 bytes 47 49 4f 50 01 02 01 00 88 00 00 00 01 00 00 00 GIOP............ 03 00 00 00 00 00 cd cd 1b 00 00 00 14 01 0f 00 ................ 52 53 54 b6 a6 51 46 c6 3e 05 00 00 00 00 00 01 RST..QF.>....... 00 00 00 01 00 00 00 cd 0b 00 00 00 67 65 74 5f ............get_ 73 74 72 69 6e 67 00 cd 02 00 00 00 55 05 41 54 string......U.AT 08 00 00 00 01 cd cd cd c3 00 00 00 01 00 00 00 ................ 0c 00 00 00 01 cd cd cd 01 00 01 00 09 01 01 00 ................ 78 9c db cf c0 c0 10 92 91 59 ac 00 44 89 0a 25 x........Y..D..% a9 c5 25 0a c5 25 45 99 79 e9 43 45 90 8b 01 00 ..%..%E.y.CE.... a0 b6 45 d6 ..E. TAO (3660|416) - Codeset_Manager_i::process_service_context, using tcsc = 00010001, tcsw = 00010109 Received <This is a test stringThis is a test stringThis is a test stringThis is a test stringThis is a test stringThis is a test stringThis is a test stringThis is a test stringThis is a test string > TAO (3660|416) - GIOP_Message_Base::dump_msg, send GIOP v1.2 msg, 29 data bytes, my endian, Type Reply[1] GIOP message - HEXDUMP 41 bytes 47 49 4f 50 01 02 01 01 1d 00 00 00 01 00 00 00 GIOP............ 00 00 00 00 00 00 00 00 0d 00 00 00 48 65 6c 6c ............Hell 6f 20 74 68 65 72 65 21 00 o there!. TAO (3660|416) - Transport[472]::cleanup_queue, byte_count = 41
ZIOP2 branch is reworked to use a bit in the header. Looking at the needed policies and service contexts now
the proto is reworked, using the bit works, but when we only compress the application data we have to do decompression on more places. Fragments can be compressed, or not, the request header can be splitted in several fragments, all together compression has impact on too much code. Another approach is only to leave the message header uncompressed but also compress all other data, including service context, etc. So far as I can now tell this leads to much cleaner code. This also matches the current implementation of IONA, going to try that now.
in preparation of the next round of prototyping I have done some cleanup in the GIOP handling code in TAO, several old things where in the code but not used. Also made some other interesting observations which will result in some rework in the orb core before I start with the next round of prototyping. Probably after that I will resync my branch and start with a new branch for this new proto
restarted the work on TAO. Telefonica has joined the ZIOP work and has made a new proposal (mars/08-05-01). This looks much like the prototype I made in TAO. Picking up this work again, I have several changes for the telefonica document but no framemaker available yet to edit the document.
want to merge a first version to svn head very soon. will always use zlib and ziop can be enabled with a policy but no control after it. will make it possible for users to experiment with compression and in parallel we can add the other policies that are needed, but that is more work
added dep
we are using the messaging support for sending policies around. This seems to work much better and reduces the additions we need to make
first version of ZIOP has been added to the core of TAO
Updated summary. TAO svn head now matches the beta 1 spec
fixed