moar DEVELOPING docs
This commit is contained in:
224
DEVELOPING.pod
224
DEVELOPING.pod
@@ -177,20 +177,19 @@ constraints can easily be deployed in a future release of Netdisco.
|
|||||||
|
|
||||||
The Netdisco web app is a "classic" Dancer app, using most of the bundled
|
The Netdisco web app is a "classic" Dancer app, using most of the bundled
|
||||||
features which make development really easy. Dancer is based on Ruby's Sinatra
|
features which make development really easy. Dancer is based on Ruby's Sinatra
|
||||||
framework. The theme is that many "helper" subroutines are exported into the
|
framework. Its style is for many "helper" subroutines to be exported into the
|
||||||
application namespace, to do things such as access request parameters,
|
application namespace, to do things such as access request parameters,
|
||||||
navigate around your "handler" subroutines, manage response headers, and so
|
navigate around the "handler" subroutines, manage response headers, and so on.
|
||||||
on.
|
|
||||||
|
|
||||||
Pretty much anything you want to do in a web application has been wrapped up
|
Pretty much anything you want to do in a web application has been wrapped up
|
||||||
into a neat helper routine that does the heavy lifting. This includes
|
by Dancer into a neat helper routine that does the heavy lifting. This
|
||||||
configuration and database connection management, as was discussed above.
|
includes configuration and database connection management, as was discussed
|
||||||
Also, templates can be executed and Netdisco uses the venerable
|
above. Also, templates can be executed and Netdisco uses the venerable
|
||||||
L<Template::Toolkit> engine for this.
|
L<Template::Toolkit> engine for this.
|
||||||
|
|
||||||
Like most web frameworks Dancer has a concept of "handlers" which are
|
Like most web frameworks Dancer has a concept of "handlers" which are
|
||||||
subroutines to which a specific web request is routed. For example if Netdisco
|
subroutines to which a specific web request is routed. For example if the user
|
||||||
asks for "C</device>" with some parameters the request ends up at the
|
asks for "C</device>" with some parameters, the request ends up at the
|
||||||
L<App::Netdisco::Web::Device> package's "C<get '/device'>" handler. All this
|
L<App::Netdisco::Web::Device> package's "C<get '/device'>" handler. All this
|
||||||
is done automatically by Dancer according to some simple rules. There are also
|
is done automatically by Dancer according to some simple rules. There are also
|
||||||
"wrapper" subroutines which we use to do tasks such as setting up data lookup
|
"wrapper" subroutines which we use to do tasks such as setting up data lookup
|
||||||
@@ -202,22 +201,229 @@ queries and avoid lengthy page reloads. You will see the handlers for AJAX
|
|||||||
look similar to those for GET requests but do not use Template::Toolkit
|
look similar to those for GET requests but do not use Template::Toolkit
|
||||||
templates.
|
templates.
|
||||||
|
|
||||||
|
Compared to the current Netdisco, the handler routines are very small. This is
|
||||||
|
because (a) they don't include any HTML - this is delegated to a template, and
|
||||||
|
(b) they don't include an SQL - this is delegated to DBIx::Class. Small
|
||||||
|
routines are more manageable, and easier to maintain. You'll also notice use
|
||||||
|
of modules such as L<Net::MAC> and L<NetAddr::IP::Lite> to simplify and make
|
||||||
|
more robust the handling of data.
|
||||||
|
|
||||||
=head2 Running the Web App
|
=head2 Running the Web App
|
||||||
|
|
||||||
|
Dancer apps conform to the "PSGI" standard interface for web applications,
|
||||||
|
which makes for easy deployment under many stacks such as Apache, FCGI, etc.
|
||||||
|
See L<Dancer::Deployment> for more detail.
|
||||||
|
|
||||||
|
At a minimum Netdisco can run from within its own user area as an unprivileged
|
||||||
|
user, and ships with a simple web server engine (see the user docs for
|
||||||
|
instructions). The C<netdisco-web> script uses L<Daemon::Control> to daemonize
|
||||||
|
this simple web server so you can fire-and-forget the Netdisco web app without
|
||||||
|
much trouble at all. This script in turn calls C<netdisco-web-fg> which is the
|
||||||
|
real Dancer application, that runs in the foreground if called on its own.
|
||||||
|
|
||||||
|
All web app code lives below L<App::Netdisco::Web>, but there are also some
|
||||||
|
helper routines in L<App::Netdisco::Util::Web> (for example sorting device
|
||||||
|
port names).
|
||||||
|
|
||||||
=head2 Authentication
|
=head2 Authentication
|
||||||
|
|
||||||
|
Dancer includes (of course) good session management using cookies and a memory
|
||||||
|
database. You should change this to a disk database if using a proper forking
|
||||||
|
web server installation so that sessions are available to all instances.
|
||||||
|
|
||||||
|
Session and authentication code lives in L<App::Netdisco::Web::AuthN>. It is
|
||||||
|
fully backwards compatible with the existing Netdisco user management, making
|
||||||
|
use of the database users and their MD5 passwords.
|
||||||
|
|
||||||
|
There is also support for unauthenticated access to the web app (for instance
|
||||||
|
if you have some kind of external authentication, or simply trust everyone).
|
||||||
|
|
||||||
=head2 Templates
|
=head2 Templates
|
||||||
|
|
||||||
|
In the C<share/views> folder of this distribution you'll find all the
|
||||||
|
Template::Toolkit template files, with C<.tt> extensions. Dancer first loads
|
||||||
|
C<share/views/layouts/main.tt> which is the main page wrapper, that has the HTML
|
||||||
|
header and so on. It then loads other templates for sections of the page body.
|
||||||
|
This is a typical Template::Toolkit "wrapper" configuration, as noted by the
|
||||||
|
C<[% content %]> call within C<main.tt> that loads the template you actually
|
||||||
|
specified in your Dancer handler.
|
||||||
|
|
||||||
|
All templates (and Javascript and Stylesheets) are shipped in the Dancer
|
||||||
|
distribution and located automatically by the application (using the
|
||||||
|
environment variables which L<App::Netdisco> set up). The user doesn't have to
|
||||||
|
copy or install any files.
|
||||||
|
|
||||||
|
There's a template for the homepage called C<index.tt>, then separate
|
||||||
|
templates for searching, displaying device details, and showing inventory.
|
||||||
|
These are, pretty much, all that Netdisco ever does.
|
||||||
|
|
||||||
|
Each of these pages is designed in a deliberately similar way, with re-used
|
||||||
|
features. They each can have a "sidebar" with a search form (or additional
|
||||||
|
search parameters). They also can have a tabbed interface for sub-topics.
|
||||||
|
|
||||||
|
Here's where it gets interesting. Up till now the page content has been your
|
||||||
|
typical synchronous page load (a single page comprised of many templates) in
|
||||||
|
response to a GET request. However the content of the tabs is not within this.
|
||||||
|
Each tab has its content dynamically retrieved via an AJAX request back to the
|
||||||
|
web application. Javscript triggers this automatically on page load.
|
||||||
|
|
||||||
|
This feature allows the user to search and search again, each time refreshing
|
||||||
|
the data they see in the tab but without reloading the complete page with all
|
||||||
|
its static furniture. AJAX can, of course, return any MIME type, not only JSON
|
||||||
|
but also HTML content as in this case. The templates for the tabs are
|
||||||
|
organised below C<share/views/ajax/...> in the distribution.
|
||||||
|
|
||||||
|
=head2 Stylesheets
|
||||||
|
|
||||||
|
The main style for Netdisco uses Twitter Bootstrap, which is a stylish modern
|
||||||
|
library of styles and javascript used on many websites. It does a lot of heavy
|
||||||
|
lifting, providing simple CSS classes for all of the standard web page
|
||||||
|
furniture (forms, tables, etc). Check out the documetation at the Twitter
|
||||||
|
Bootstrap web site for more information.
|
||||||
|
|
||||||
|
These stylesheets are of course customised with our own C<netdisco.css>. We
|
||||||
|
try to name all CSS classes with a prefix "C<nd_>" so as to be distinct from
|
||||||
|
Twitter Bootstrap and any other active styles.
|
||||||
|
|
||||||
|
All stylesheets are located in the C<share/public/css> folder of the
|
||||||
|
distribution and, like the templates, are automatically located and served by
|
||||||
|
the Netdisco application. You can also choose to serve this content statically
|
||||||
|
via Apache/etc for high traffic sites.
|
||||||
|
|
||||||
|
Although Twitter Bootstrap ships with its own set of icons, we use an
|
||||||
|
alternative library called Fontawesome. This plugs in easily to Bootstrap and
|
||||||
|
provides a wider range of scaleable vectored icons which are easy to use.
|
||||||
|
|
||||||
=head2 Javascript
|
=head2 Javascript
|
||||||
|
|
||||||
|
Of course many parts of the Netdisco site use Javascript, beginning with
|
||||||
|
retrieving the page tab content itself. The standard library in use is jQuery,
|
||||||
|
and the latest version is shipped with this distribution.
|
||||||
|
|
||||||
|
Many parts of the Netdisco site have small Javscript routines. The code for
|
||||||
|
these, using jQuery as mentioned, lives in two places. The main C<netdisco.js>
|
||||||
|
file is loaded once in the page HTML header, and lives in
|
||||||
|
C<share/public/javascripts/netdisco.js>. There's also a
|
||||||
|
C<netdisco_portcontrol.js> which is included only if the current user has Port
|
||||||
|
Control rights.
|
||||||
|
|
||||||
|
Netdisco also has Javascript routines specific to the device search or device
|
||||||
|
details pages, and these files are located in C<share/views/js/...> because
|
||||||
|
they're loaded within the page body by the templates. These files contain a
|
||||||
|
function C<inner_view_processing> which is called each time AJAX delivers new
|
||||||
|
content into a tab in the page (think of it like a callback, perhaps).
|
||||||
|
|
||||||
|
Also in the C<share/public/javascripts/...> folder are the other public
|
||||||
|
libraries loaded by the Netdisco application:
|
||||||
|
|
||||||
|
The Toastr library is used for "Growl"-like notifications which appear in the
|
||||||
|
corner of the web browser and then fade away. These notify the user of
|
||||||
|
successful background job submission, and jos results.
|
||||||
|
|
||||||
|
The d3 library is a graphics toolkit used to display the NetMap feature. This
|
||||||
|
works differently from the old Netdisco in that everything is generated
|
||||||
|
on-the-fly using SQL queries (C<DeviceLinks> resultset) and this d3 library
|
||||||
|
for rendering.
|
||||||
|
|
||||||
|
Finally Twitter Bootstrap also ships with a toolkit of helpful Javascript
|
||||||
|
driven features such as the tooltips and collapsers.
|
||||||
|
|
||||||
|
|
||||||
=head1 Job Daemon
|
=head1 Job Daemon
|
||||||
|
|
||||||
|
The old Netdisco has a job control daemon which processes "port control"
|
||||||
|
actions and also manual requests for device polling. The new Netdisco also has
|
||||||
|
a daemon, although it is a true separate process and set of libraries from the
|
||||||
|
web application. However, it still makes use of the Dancer configuration and
|
||||||
|
database connection management features mentioned above.
|
||||||
|
|
||||||
|
The job daemon is backwards compatible with the old Netdisco database job
|
||||||
|
requests table, although it doesn't yet log results in the same way. Most
|
||||||
|
important, it cannot yet poll any devices for discovery or macsuck/arpnip,
|
||||||
|
although that's next on the list!
|
||||||
|
|
||||||
|
All code for the job daemon lives under the L<App::Netdisco::Daemon> namespace
|
||||||
|
and like the rest of Netdisco is broken down into manageable chunks.
|
||||||
|
|
||||||
|
=head2 Running the Job Daemon
|
||||||
|
|
||||||
|
Like the web application, the job daemon is fully self contained and runs via
|
||||||
|
two simple scripts shipped with the distribution - one for foreground and one
|
||||||
|
for background execution (see the user docs for instructions).
|
||||||
|
|
||||||
|
The C<netdisco-daemon> script uses L<Daemon::Control> to daemonize so you can
|
||||||
|
fire-and-forget the Netdisco job daemon without much trouble at all. This
|
||||||
|
script in turn calls C<netdisco-daemon-fg> which is the real application, that
|
||||||
|
runs in the foreground if called on its own.
|
||||||
|
|
||||||
|
=head2 Daemon Engineering
|
||||||
|
|
||||||
|
The job daemon is based on the L<MCE> library, which handles the forking and
|
||||||
|
management of child processes doing the actual work. This actually runs in the
|
||||||
|
foreground unless wrapped with Daemon::Control, as mentioned above. MCE
|
||||||
|
handles three flavours of "worker" for different tasks.
|
||||||
|
|
||||||
|
One goal that we had designing the daemon was that sites should be able to run
|
||||||
|
many instances on different servers, with different processing capacities.
|
||||||
|
This is both to take advantage of more processor capability, but also to deal
|
||||||
|
with security zones where you might only be able to manage a subset of devices
|
||||||
|
from certain locations. Netdisco has always coped well with this via its
|
||||||
|
C<discover_*> and similar configuration, and the separate poller process.
|
||||||
|
|
||||||
|
So, the single Manager "worker" in the daemon is responsible for contacting
|
||||||
|
the central Netdisco database and booking out jobs which it's able to service
|
||||||
|
according to the local configuration settings. Jobs are "locked" in the
|
||||||
|
central queue and then copied to a local job queue within the daemon.
|
||||||
|
|
||||||
|
Along with the Manager we start zero or more of two other types of worker.
|
||||||
|
Some jobs such as port control are "interactive" and the user typically wants
|
||||||
|
quick feedback on the results. Others such as polling are background tasks
|
||||||
|
which can take more time and are less schedule sensitive. So as not to starve
|
||||||
|
the "interactive" jobs of workers we have two types of worker.
|
||||||
|
|
||||||
|
The Interactive worker picks jobs from the local job queue relating to device
|
||||||
|
and port reconfiguration only. It submits results directly back to the central
|
||||||
|
Netdisco database.
|
||||||
|
|
||||||
|
The Poller worker (is not yet written!) and similarly picks job from the local
|
||||||
|
queue, this time relating to device discovery and polling.
|
||||||
|
|
||||||
|
There is support in the daemon for the workers to pick more than one job at a
|
||||||
|
time from the local queue, in case we decide this is worth doing. However the
|
||||||
|
Manager won't ever book out more jobs from the central Netdisco job queue than
|
||||||
|
it has workers available (so as not to hog jobs for itself against other
|
||||||
|
daemons on other servers). The user is free to configure the number of
|
||||||
|
Interactive and Poller workers in their C<config.yml> file (zero or more of
|
||||||
|
each).
|
||||||
|
|
||||||
=head2 SNMP::Info
|
=head2 SNMP::Info
|
||||||
|
|
||||||
|
The daemon obviously needs to use L<SNMP::Info> for device control. All the
|
||||||
|
code for this has been factored out into the L<App::Netdisco::Util> namespace.
|
||||||
|
|
||||||
|
The L<App::Netdisco::Util::Connect> package provides for the creation of
|
||||||
|
SNMP::Info objects along with connection tests. So far, SNMPv3 is not
|
||||||
|
supported. To enable trace logging of the SNMP::Info object simply set the
|
||||||
|
C<INFO_TRACE> environment variable to a true value. The Connect library also
|
||||||
|
provides routines to map interface and PoE IDs.
|
||||||
|
|
||||||
|
Configuration for SNMP::Info comes from the YAML files, of course. This means
|
||||||
|
that our C<mibhome> and C<mibdirs> settings are now in YAML format. In
|
||||||
|
particular, the C<mibdirs> list is a real list within the configuration.
|
||||||
|
|
||||||
|
Other libraries will be added to this namespace in due course, as we add more
|
||||||
|
functionality to the Job Daemon.
|
||||||
|
|
||||||
=head2 DBIx::Class Layer
|
=head2 DBIx::Class Layer
|
||||||
|
|
||||||
=head2 Running the Job Daemon
|
The local job queue for each Job Daemon is actually an SQLite database running
|
||||||
|
in memory. This makes the queue management code a little more elegant. The
|
||||||
|
schema for this is of course DBIx::Class using Dancer connection management,
|
||||||
|
and lives in L<App::Netdisco::Daemon::DB>.
|
||||||
|
|
||||||
|
There is currently only one table, the port control job queue, in
|
||||||
|
L<App::Netdisco::Daemon::DB::Result::Admin>. It's likely this name will change
|
||||||
|
in the future.
|
||||||
|
|
||||||
=cut
|
=cut
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user