Revision history for Perl extension POE::Component::SNMP. -*-text-*-

0.01  Mon Apr 21 13:28:31 2003
        - original version; created by h2xs 1.22 with options
                -AX -n POE::Component::SNMP

0.90  Mon Jun  7 18:44:00 2004
        - Module inherited by Rob Bloodgood, from the previous
          maintainer, Todd Caine.

        - added POE::Component::SNMP::Dispatcher module, to subclass
          Net::SNMP::Dispatcher such that all socket and scheduling
          operations were processed by POE's event loop.

        - updated the callback variables so that the hostname and
          session alias of the SNMP object queried are returned to the
          callback event.

        - made write tests optional during install, so that the module
          would install correctly even where a writeable host was
          unavailable.
        
0.92  Mon Sep  6 13:19:53 2004
        - Updated to be compatible with new Net::SNMP release 5.0.0

0.93  Thu Aug 11 11:57:29 2005
        - Updated POE session constructor from the deprecated new() to
          the new create() method.

        - Removed a bunch of commented, deprecated code that was still
          a holdover from the 0.01

0.94  Thu Mar 30 14:43:19 2006
        - Reorganized the module distribution to use a lib/ dir, cuz
          MakeMaker was being dumb.

        - Updated Makefile.PL to remove the config file on 'make
          realclean'

        - Tweaked the docs.

        - Altered the sample script in the SYNOPSIS to actually work.

        - Removed remaining commented references to deprecated code.

        - Followed a suggestion from Curtis J. Coleman to re-order
          component shutdown so that finish() worked correctly.

        - Lots of minor cleanups in the text

        - At LAST: removed the duplicate methods to account for the
          difference between Net::SNMP 4.x and 5.x.  Now this is
          achieved by aliasing the functions that need to be
          duplicated in the symbol table.

0.95  Sat Apr  8 14:46:54 PDT 2006
        - Collapsed all the identical method calls into a dispatch
          mechanism around a single method call template.

        - VASTLY improved error handling.

        - Added the snmp method (eg get, set, walk) and request args
          to the request packet returned to callbacks.

        - Reorganized Dispatcher.pm, trimmed old commented
          code. Removed unneeded debugging statements.

	- trimmed unnecessary parameters from calls to
          __invoke_callback().

        - added new tests for various aspects of error handling.

0.96  Mon Apr 10 01:29:40 PDT 2006
	- re-ordered the hostname and community checks so that the
	  fatal "hostname is required" is thrown BEFORE the warning
	  "using default community".

	- added new tests for error conditions and parameter errors

	- changed the return values to always return something sane,
          e.g. a string or a ref, not a string or a ref or maybe an
          undef, THEN a string!  The only reason I didn't change this
          before is that I mistakenly thought this weird error
          structure came directly from Net::SNMP, but it turns out I
          just inherited it. <sigh>

	- for compatibility, continue to return the $snmp_object->error
	  value in the second slot, since the last change alters the API.

	- Also, changed the original args returned to callbacks from
          being a listref to a simple list, in the same way they
          came in.

	- Unified style in docs.  Cleaned up internal and external POD
          links.

0.97  Mon Apr 10 13:14:01 PDT 2006
        - Fixed some formatting tweaks in the docs.

        - Renamed the function snmp_last_error_message to snmp_errmsg,
          to correspond the the component request 'errmsg', like the
          other method names.

        - DANG!  Left in some debugging references to YAML and Spiffy
          that I meant to comment out. Which breaks systems that don't
          have those modules installed.  Grr.  Fixed.

        - Also pulled references to YAML from the sample script... not
          everybody has YAML.

0.98  Wed Apr 12 20:02:12 PDT 2006
        - Improved dispatcher tracing by making all calls consistently
          trace their execution.  Use Net::SNMP debug functions to
          report our activities when the debug flag to trace the
          Dispatcher is set in the constructor.  Gives nice,
          consistent trace output.

        - Started work to remove the requirement that each host have a
          different -localport parameter.  I chased it down.  The
          fundemental question is, how is it possible for Net::SNMP to
          reuse the same socket for every remote host, without parsing
          the internals of the packet and matching them to a "replies
          expected" table?  The answer is, it is not.  Net::SNMP
          dispatches all queries in the order received, serially.  The
          workaround we used is correct.  Note to self: NOW STOP
          WONDERING IF THIS IS POSSIBLE!

        - Accidentally optimized the whole module by a factor of at
          least 3x by copying Net::SNMP::Dispatcher's send_pdu() into
          PoCo::SNMP::Dispatcher.  See notes on send_pdu().

        - Changed __schedule_event so that events with no delay simply
          invoke their callback args instead of post()ing to
          __invoke_callback.  This saves me a POE state call.

        - Updated _send_pdu to return TRUE, just like the Net::SNMP
          version does.

0.99
	- Changed send_pdu() optimization. Instead of being a copy of
          the N:S:D's, we simply change the function reference
          involved, with local(), before invoking SUPER::send_pdu().

        - Changed __send_next_pdu and __pdu_sent around... the 2nd
          half of __send_next_pdu belonged logically in __pdu_sent.
          This saves me a POE state call and is cleaner.

        - Renamed __send_next_pdu to __dispatch_pdu.

        - Discovered that select_read() can take additional args to
          pass back to the event triggered when a handle goes live.
          Duuhh.  What this means is, I no longer have to dig into
          Net::SNMP to find out where they stashed the callback,
          indexed by the fileno of the socket.  Instead I can simply
          pass and invoke the supplied callback.  This changed
          register(), __listen, and __socket_callback.

	- Discovered that extra args support for select_read() wasn't
          added until 0.32. <sigh> Coded it back, only this time
          without mucking around with private attributes of my parent
          class.

	- Tweaked whitespace so stuff looks consistent.

	- Copied the sanity check from Net::SNMP that loads and
          verifies the dispatcher instance.  Now P:C:S:Dispatcher
          doesn't muck around with Net::SNMP's package space, just
          subclasses it.  Instead, P:C:S loads the Dispatcher in a
          BEGIN block, just like Net::SNMP.  THEN it mucks with
          Net::SNMP's namespace. <evil grin>

	- Replaced references to $self in the constructor with $this,
          following Net::SNMP's convention and the rest of the module.

	- Removed alot of folds.  I had folds around every function,
          now I just have them around major sections.

	- Started working on the edge case where somebody is dumb
          enough to call finish() while data is still pending.	

	- Updated constructor to accept all forms of -alias, alias,
          Alias, and the same for -hostname.

	- Removed the check for a valid varbindlist in the error
          handler of the snmp_request().  Just return the error.

	- Changed tracking of _current_pdu from a reference to the
          callback args to a simple flag.  All it needs to do is
          *exist*.

        - Triple-checked that the reference changing done in
          send_pdu() was indeed persisting for subsequent
          invocations of _send_pdu().

        - Adusted the debug output to have a more verbose mode
          available.

        - More work on the finish() cleanup... now we delete pending
          requests but allow the current request to complete.

	- Altered socket listen/unlisten operations to use
          select_pause_read() and select_resume_read(), because docs
          say they are more lightweight.  Initial tests show reduction
          of approx. 20% in system loadavg with loadavg_multi_snmp.

	- moved the socket ops to two new private methods,
          _watch_handle() and _unwatch_handle().  These functions
          maintain the reference counting of who has what handle,
          without any of the rest of the module having to count any
          part of it.  that's good. :)

	- had to adjust _unwatch_handle() to NOT stop listening when
          there are replies due, in __clean_pending

	- put the folds back in, not quite as many before tho.  it's
          easier to go from block to block that way, and only have the
          functions unfolded that I'm working on.

	- moved tracking of current and pending requests from the
          session heap to the dispatcher object, so that the info can
          be retrieved in POE or non-POE space.

	- added the private method _pending_pdu_count(), based on
          the last change.

        - renamed __pdu_sent to __dispatch_pending_pdu

	- use _pending_pdu_count() check to skip invoking
          __dispatch_pending_pdu unless there is actually anything
          pending.  this saves me a POE state call for most situations.
	
	- added accessors for current and pending pdus, instead of
          getting directly into the heap hash or object.

	- tried to change __clear_pending from an event to an object
          method, but it hangs when I do that.

	- way, way more tests.  almost rewrote 40_set from
          scratch... now instead of just blindly setting something, it
          reads it, sets something else, verifies, then puts it back
          and verifies.  Much more robust.

        - turns out that a better hook for subclassing than schedule()
          is _event_insert().  By using it instead, we get A) the same
          $event reference that cancel() receives, and B) a method
          name that does *not* change between 4.x and 5.x.  The logic
          for cancel() stays the same.

	- commented references to schedule/_schedule in the
          compatibility section.

        - changed _event_insert() so that if the time to invoke the
          callback is immediate, it is invoked right then, *instead*
          of calling __schedule.  This saves me another POE state
          call, woo-hoo!

        - created a global debug flag in the config file so that I can
          turn on and off debug output in *all* tests with one change.

        - followed to the suggestion of Curtis J. Coleman to add the
          ability to set arbitrary callback arguments to be returned
          to callback events along with ordinary response data.
          created tests for it. actually, I put in a couple of lines
          of code and then wrote the tests... and then actually put in
          the feature. :) tests are cool.

        - changed a coupla stray call()'s in the tests to posts().

        - added tests and logic for localport conflicts... my previous
          code didn't work, and I ran into this error: "Failed to bind
          UDP/IPv4 socket: Address already in use at ~/bin/wackastat
          line 38", followed by "POE::Kernel's run() method was never
          called."

        - added an _arg_scan() to flexibly fetch values from a *list*
          of key/value pairs.

        - finally tracked down and eliminated the last stray reference
          that was holding our sessions open when we called 'finish'
          with requests still in the pipe!!!!  It turns out the
          $MESSAGE_PROCESSING subsystem was holding a cached copy of
          our request, including its callback.

        - put an error check in __dispatch_pending_pdu based on an
          error report from Curtis Coleman

1.00 Mon Apr 24 15:55:23 PDT 2006
        - release

1.01 Mon Apr 24 17:41:08 PDT 2006
        - already got an error report from Curtis
          Coleman.. __socket_callback ended up seeing an empty
          callback.  Fix by deferring unless there is a current callback.

1.02 Tue Apr 25 11:38:00 PDT 2006
        - CPAN automated testing indicates not being able to correctly
          locate TestPCS.pm.  Turns out I forgot to put it into the
          MANIFEST, so it wasn't bundled in the distribution!

        - Turns out my tests were anticipating the wealth of results
          available from the Net-SNMP daemon, which is not delivered
          from, for example, an Alcatel DSLAM.  With the invaluable
          assistance of Curtis Coleman again, as well as a few others
          in #PoE, got the tests straightened out.  Thanks guys!

1.03 Fri Jun  2 18:51:19 PDT 2006
        - small doc tweaks

        - micro-optimized _pending_pdu_count by skipping exists() and
          just using ref().

        - added a check to __schedule_event to execute immediately if
          for some strange reason the time of execution has already
          arrived and yet we're still invoking __schedule.

        - fixed a logic error in the -localport initializer that was
          preventing retries if the random port chosen was already in
          use.  Thanks to Curtis Coleman for the spot.

1.04 Tue Jun  6 14:35:30 PDT 2006
        - my fix was broken.  fixed for real this time.  thanks to
          Curtis AGAIN!

1.05 Fri Jun 16 13:00:40 PDT 2006
        - turned off $VERBOSE in Dispatcher.pm... even when debug
          messages weren't being shown, VERBOSE adds some extra
          processing that is unnecessary (unless, of course, you're
          developing PoCo::SNMP :).

        - my assumptions about having a singleton dispatcher session
          fell apart when a user tried to run the POE kernel more than
          once... after the first run, the session is destroyed,
          causing subsequent runs to fail.  added destructor logic to
          the dispatcher's _stop event, and constructor logic to the
          main component's create() method.  Now it functions properly
          if run more than once.  Thanks to Nigel Bowden for the spot.

1.06 Tue Sep 12 00:43:42 PDT 2006
        - finish() behavior was broken... calling finish() for one
          session finished (well, broke) the entire component,
          PERMANENTLY.  oops.  removed all references to the internal
          "_abort" flag.  now the caveat is, if you finish() using
          POE::Kernel->call() in the same event that you've made
          requests, the program will hang because there will be stray
          events.  use post(), or yield() instead.  Thanks to Kenny
          Flegal for the spot *and* the test case.

	- cleaned up the -localport logic a little bit... added a
          do..while() where I had a while() plus the same line of code
          twice.

        - added a 'trap2c' method a version or two ago... but forgot
          to add it to the states list in the constructor for the
          PoCo::SNMP session.  This meant that anybody trying to use
          it would silently fail.  oops.  same for 'callback_args'.
          oops twice.

        - did a little bit of refactoring in Dispatcher.pm to make
          identifiers "tell the story" a little better, cuz on a
          re-read of the logic it wasn't "instantly clear" what I'd
          been trying to do. __dispatch_pdu and __dispatch_pending_pdu
          read more clearly now.

        - made my debugging functions a little better at no-op'ing
          when not debugging, resulting in a small improvement in
          "make test" times.

1.07 Wed Sep 27 16:30:17 PDT 2006
        - added 71_multi_autherror.t to test SNMP v2c authentication
          errors.  Net::SNMP returns 'no response'.

        - changed the call to POE::Kernel->post() to yield() in
          deregister(), saving the overhead of resolving the session
          since it is always called within the Dispatcher session.

        - also changed register() and schedule() in the Dispatcher to
          use POE::Kernel->call() instead of post(), because they
          doesn't hurt concurrency there, all they do is select_read()
          or alarm_set() then return.

        - removed an extraneous localport() accessor from SNMP.pm.

        - re-enabled the "duplicate session" check in the constructor
          because POE now correctly allows the check under
          ASSERT_DEFAULT.

        - Finally added folds to SNMP.pm.

        - Minor changes to the documentation.

        - Removed an unused variable from __clear_pending().

1.10 Fri Jul 25 01:39:08 PDT 2008
        - Added a Test::Pod pod.t to increase module Kwalitee.  *NOT*
          adding Test::Pod::Coverage stuff because my "coverage" isn't
          in functions, but in states.  Plus, Dispatcher doesn't have
          or need docs.  So there.

        - altered/renamed _get_next_pending_pdu(), returning a scalar
          to _get_next_pending_pdu_args, returning an array, to unify
          __dispatch_pdu and __dispatch_pending_pdu

        - merged the __dispatch_pending_pdu event completely into
          __dispatch_pdu, by making deregister fetch the pending args,
          since the only difference between them was where they got
          the args for the SUPER::_send_pdu() call.

        - added clarification to the docs WRT concurrency.

        - fixed a small bug in the error handling for callback_args.

        - POE_ASSERT_DATA still throws a fatal error when
          alias_resolve() fails (up until 0.95), so updated the check
          to match this "minimum" condition.

        - updated the docs to have sample usages of each of the
          request types.

        - tweaked the localport conflict detection slightly.  Added a
          test for it.

        - Andrew Hoying sent in a patch adding the getentries method,
          along with POD *and* a test case... woo-hoo!  He reports
          that switching some of his code from walk to getentries
          significantly cut down on the amount of packets sent over
          the network.

        - renamed (un)?watch_transport to (un)?watch_socket.  It's
          closer to what I really mean, and I've started work on
          PoCo::SNMP::Session, which will have such a similar
          structure to this module that I want to unify the
          differences as far as possible.

        - attempted to alter socket watch/unwatch behavior to leave
          the socket "hot" in between requests.  This saves 2 method
          calls and 2 kernel calls per request.  However, it's not
          quite right, so it's currently disabled

        - fixed a bug where SNMPv3 requests that fail could let me wind
          up in __clear_pending without a valid socket to unwatch/clear.

        - fixed a bug where I checked _pending_pdu without passing a
          (mandatory) fileno parameter.  Oops.

        - got the "hot" socket logic right.  I kinda cheated... I just
          skipped right past pausing/resuming select_read on the sockets.



        - finally got the tests right.  I hope.