| Ekahau documentation | Contained in the Ekahau distribution. |
Ekahau - Synchronous interface to Ekahau location sensing system
The Ekahau class provides a straightforward synchronous interface
to the Ekahau location sensing system's YAX protocol. The YAX
protocol itself is asynchronous, so this module tries its best to hide
that from you.
This class inherits from Ekahau::Base, and you can use methods from that class. An alternative, event-driven interface is provided by Ekahau::Events.
This class implements methods for querying the Ekahau Positioning
Engine, and processing the responses. Each Ekahau object
represents a connection to the Ekahau server. Some methods send
queries to the server, others receive responses, and still others do
both. This class tries to hide the asynchronousness of the YAX
protocol from you.
The basic mechanism allows you to send requests, and wait for responses. If other responses unrelated to the one you're waiting for come in, they will be queued and sent to you when you ask for them. Some methods will first send a request then wait for the response, handling all of the details internally.
This classes uses Ekahau::Base::new as its constructor.
Get a list of devices currently detected on the Ekahau system. The
response is returned as a reference to a list of numeric IDs. On
error, this method will return undef.
Get the properties for the device with the given ID. Returns an Ekahau::Response object.
Get the properties of the area with the given ID. Returns an Ekahau::Response object.
Get a map of the area with the given ID. Returns an Ekahau::Response::MapImage object.
Returns a list of all logical areas, as an Ekahau::Response::AreaList object.
Start a continuous query tracking the location of the given device. An optional hash reference containing properties for the query can be given; see Ekahau::Base::start_location_track for details.
You can get results with next_location or next_track.
Stop a continuous query tracking the location of the given device.
Return the next location for any of the continuous queries started by
calls to start_location_track. The returned value will be an
Ekahau::Response object, or undef on error.
Start a continuous query tracking the area where the given device is located. An optional hash reference containing properties for the query can be given; see Ekahau::Base::start_area_track for details.
You can get results with next_area or next_track.
Stop a continuous query tracking the area of the given device.
Return the next area for any of the continuous queries started by
calls to start_area_track. The returned value will be an
Ekahau::Response object, or undef on error.
Start tracking both the location and the area of the given device, by calling the start_location_track and start_area_track methods.
An optional hash reference containing properties for the query can be given. These properties will be passed on to both queries; if you need different properties for the two queries call the methods directly.
Results can be obtained by calling the next_track method.
Stop tracking both the location and the area of the given device, by calling the stop_location_track and stop_area_track methods.
Return the next location or area for any of the continuous queries
started by calls to start_track, start_location_track, or
start_area_track. The returned value will be an
Ekahau::Response object, or undef on error.
Scott Gifford <gifford@umich.edu>, <sgifford@suspectclass.com>
Copyright (C) 2005 The Regents of the University of Michigan.
See the file LICENSE included with the distribution for license information.
| Ekahau documentation | Contained in the Ekahau distribution. |
package Ekahau; use base 'Ekahau::Base'; our $VERSION = $Ekahau::Base::VERSION; # Written by Scott Gifford <gifford@umich.edu> # Copyright (C) 2004 The Regents of the University of Michigan. # See the file LICENSE included with the distribution for license # information. use warnings; use strict;
sub getresponse { my $self = shift; my ($tags,$cmds)=@_; my $tagmap = (($tags and @$tags) ? { map { $_ => 1 } @$tags } : undef); my $cmdmap = (($cmds and @$cmds) ? { map { $_ => 1 } @$cmds } : undef); # See if we have something on the queue already if ($self->{_q}) { foreach my $i (0..$#{$self->{_q}}) { if (_respmatch($self->{_q}[$i],$tagmap,$cmdmap)) { return splice(@{$self->{_q}},$i,1); } } } # Wait until we get something, or the timeout expires. my $started = time; while(1) { while (my $resp = $self->getpending) { return $resp if (_respmatch($resp,$tagmap,$cmdmap)); push @{$self->{_q}},$resp; } # See if we timed out. $self->can_read($self->{_timeout}?($self->{_timeout}-(time-$started)):0) or return undef; $self->readsome() or return undef; } }
sub gettaggedresponse { my $self = shift; $self->getresponse([@_]); } sub _respmatch { my($resp,$tagmap,$cmdmap) = @_; return ((!$tagmap and !$cmdmap) or (!$cmdmap and $resp->{tag} and $tagmap->{$resp->{tag}}) or (!$tagmap and $resp->{cmd} and $cmdmap->{$resp->{cmd}}) or ($resp->{cmd} and $resp->{tag} and $tagmap->{$resp->{tag}} and $cmdmap->{$resp->{cmd}})); }
sub get_device_list { my $self = shift; my $tag = $self->request_device_list or return undef; my $resp = $self->gettaggedresponse($tag) or return undef; if ($resp->error) { return $self->reterr("GET_DEVICE_LIST failed: ".$resp->error_description); } elsif ($resp->type ne 'DeviceList') { return $self->reterr("GET_DEVICE_LIST failed: unexpected response!"); } return [$resp->devices]; }
sub get_device_properties { my $self = shift; my $tag = $self->request_device_properties(@_) or return undef; my $resp = $self->gettaggedresponse($tag) or return undef; if ($resp->error) { return $self->reterr("GET_DEVICE_PROPERTIES failed: ".$resp->error_description); } elsif ($resp->type ne 'DeviceProperties') { return $self->reterr("GET_DEVICE_PROPERTIES failed: unexpected response of type ".$resp->type."!"); } $resp; }
sub get_location_context { my $self = shift; my $tag = $self->request_location_context(@_) or return undef; my $resp = $self->gettaggedresponse($tag) or return undef; if ($resp->error) { $self->{err} = "GET_CONTEXT failed: ".$resp->error_description; return undef; } elsif ($resp->type ne 'LocationContext') { $self->{err} = "GET_CONTEXT failed: unexpected response!"; return undef; } $resp; }
sub get_map_image { my $self = shift; my $tag = $self->request_map_image(@_) or return undef; my $resp = $self->gettaggedresponse($tag) or return undef; if ($resp->error) { return $self->reterr("GET_MAP failed: ".$resp->error_description); } elsif ($resp->type ne 'MapImage') { return $self->reterr("GET_MAP failed: unexpected response!"); } $resp; }
sub get_all_areas { my $self = shift; my $tag = $self->request_all_areas(@_) or return undef; my $resp = $self->gettaggedresponse($tag) or return undef; if ($resp->error) { return $self->reterr("GET_LOGICAL_AREAS failed: ".$resp->error_description); } elsif ($resp->type ne 'AreaList') { return $self->reterr("GET_LOGICAL_AREAS failed: unexpected response!"); } $resp; }
sub start_location_track { my $self = shift; my %p = ref $_[0] ? %{ (shift) } : (); $p{Tag} = 'LOC'; $self->SUPER::start_location_track(\%p,@_); }
sub stop_location_track { my $self = shift; my $tag = $self->request_stop_location_track(@_) or return undef; my $resp = $self->gettaggedresponse($tag) or return undef; if ($resp->error) { return $self->reterr("STOP_LOCATION_TRACK failed: ".$resp->error_description); } elsif ($resp->type ne 'StopLocationTrackOK') { return $self->reterr("STOP_LOCATION_TRACK failed: unexpected response '".$resp->type."!"); } $resp; }
sub next_location { my $self = shift; my $r = $self->gettaggedresponse('LOC') or return undef; if ($r->{cmd} ne 'LOCATION_ESTIMATE') { return $self->reterr("LOCATION_ESTIMATE failed: $r->{cmd} @{$r->{args}}"); } return $r; }
sub start_area_track { my $self = shift; my %p = ref $_[0] ? %{ (shift) } : (); $p{Tag} = 'AREA'; $self->SUPER::start_area_track(\%p,@_); }
sub stop_area_track { my $self = shift; my $tag = $self->request_stop_area_track(@_) or return undef; my $resp = $self->gettaggedresponse($tag) or return undef; if ($resp->error) { return $self->reterr("STOP_AREA_TRACK failed: ".$resp->error_description); } elsif ($resp->type ne 'StopAreaTrackOK') { return $self->reterr("STOP_AREA_TRACK failed: unexpected response '".$resp->type."'!"); } $resp; }
sub next_area { my $self = shift; my $r = $self->gettaggedresponse('AREA'); if ($r->{cmd} ne 'AREA_ESTIMATE') { return $self->reterr("AREA_ESTIMATE failed: $r->{cmd} @{$r->{args}}"); } return $r; }
sub start_track { my $self = shift; # Use temp variables so we can return the correct result, # but still make sure we start both. my $e1 = $self->start_location_track(@_); my $e2 = $self->start_area_track(@_); return $e1 and $e2; }
sub stop_track { my $self = shift; # Use temp variables so we can return the correct result, # but still make sure we start both. my $e1 = $self->stop_location_track(@_); my $e2 = $self->stop_area_track(@_); return $e1 and $e2; }
sub next_track { my $self = shift; my(%p)=@_; return $self->gettaggedresponse('LOC','AREA'); } 1;