initial commit of file from CVS for e-smith-lib on Wed 12 Jul 08:58:46 BST 2023
This commit is contained in:
parent
6d7e97ea37
commit
a527984040
4
.gitignore
vendored
Normal file
4
.gitignore
vendored
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
*.rpm
|
||||||
|
*.log
|
||||||
|
*spec-20*
|
||||||
|
*.tar.xz
|
21
Makefile
Normal file
21
Makefile
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
# Makefile for source rpm: e-smith-lib
|
||||||
|
# $Id: Makefile,v 1.1 2016/02/05 22:44:50 stephdl Exp $
|
||||||
|
NAME := e-smith-lib
|
||||||
|
SPECFILE = $(firstword $(wildcard *.spec))
|
||||||
|
|
||||||
|
define find-makefile-common
|
||||||
|
for d in common ../common ../../common ; do if [ -f $$d/Makefile.common ] ; then if [ -f $$d/CVS/Root -a -w $$/Makefile.common ] ; then cd $$d ; cvs -Q update ; fi ; echo "$$d/Makefile.common" ; break ; fi ; done
|
||||||
|
endef
|
||||||
|
|
||||||
|
MAKEFILE_COMMON := $(shell $(find-makefile-common))
|
||||||
|
|
||||||
|
ifeq ($(MAKEFILE_COMMON),)
|
||||||
|
# attept a checkout
|
||||||
|
define checkout-makefile-common
|
||||||
|
test -f CVS/Root && { cvs -Q -d $$(cat CVS/Root) checkout common && echo "common/Makefile.common" ; } || { echo "ERROR: I can't figure out how to checkout the 'common' module." ; exit -1 ; } >&2
|
||||||
|
endef
|
||||||
|
|
||||||
|
MAKEFILE_COMMON := $(shell $(checkout-makefile-common))
|
||||||
|
endif
|
||||||
|
|
||||||
|
include $(MAKEFILE_COMMON)
|
16
README.md
16
README.md
@ -1,3 +1,17 @@
|
|||||||
# e-smith-lib
|
# <img src="https://www.koozali.org/images/koozali/Logo/Png/Koozali_logo_2016.png" width="25%" vertical="auto" style="vertical-align:bottom"> e-smith-lib
|
||||||
|
|
||||||
SMEServer Koozali developed git repo for e-smith-lib smeserver
|
SMEServer Koozali developed git repo for e-smith-lib smeserver
|
||||||
|
|
||||||
|
## Wiki
|
||||||
|
<br />https://wiki.koozali.org/
|
||||||
|
|
||||||
|
## Bugzilla
|
||||||
|
Show list of outstanding bugs: [here](https://bugs.koozali.org/buglist.cgi?component=e-smith-lib&product=SME%20Server%2010.X&query_format=advanced&limit=0&bug_status=UNCONFIRMED&bug_status=NEW&bug_status=ASSIGNED&bug_status=REOPENED&bug_status=CONFIRMED)
|
||||||
|
|
||||||
|
## Description
|
||||||
|
|
||||||
|
<br />*This description has been generated by an LLM AI system and cannot be relied on to be fully correct.*
|
||||||
|
*Once it has been checked, then this comment will be deleted*
|
||||||
|
<br />
|
||||||
|
|
||||||
|
E-smith-lib is an open source software that enables users to quickly and easily create, edit, and share digital content. The platform offers a variety of tools and features that make it easy for users to create and edit a variety of digital documents, such as text documents, audio files, video files, and photographs. It also allows users to collaborate with other users, share digital content, and manage digital workflow. E-smith-lib also features a powerful search engine that can help users locate digital content quickly. Additionally, it offers a variety of templates for quickly creating digital documents, and provides support for multiple languages and different operating systems. The software is easy to use and offers a convenient way for users to manage their digital content.
|
||||||
|
131
additional/Artistic
Normal file
131
additional/Artistic
Normal file
@ -0,0 +1,131 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
The "Artistic License"
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The intent of this document is to state the conditions under which a
|
||||||
|
Package may be copied, such that the Copyright Holder maintains some
|
||||||
|
semblance of artistic control over the development of the package,
|
||||||
|
while giving the users of the package the right to use and distribute
|
||||||
|
the Package in a more-or-less customary fashion, plus the right to make
|
||||||
|
reasonable modifications.
|
||||||
|
|
||||||
|
Definitions:
|
||||||
|
|
||||||
|
"Package" refers to the collection of files distributed by the
|
||||||
|
Copyright Holder, and derivatives of that collection of files
|
||||||
|
created through textual modification.
|
||||||
|
|
||||||
|
"Standard Version" refers to such a Package if it has not been
|
||||||
|
modified, or has been modified in accordance with the wishes
|
||||||
|
of the Copyright Holder as specified below.
|
||||||
|
|
||||||
|
"Copyright Holder" is whoever is named in the copyright or
|
||||||
|
copyrights for the package.
|
||||||
|
|
||||||
|
"You" is you, if you're thinking about copying or distributing
|
||||||
|
this Package.
|
||||||
|
|
||||||
|
"Reasonable copying fee" is whatever you can justify on the
|
||||||
|
basis of media cost, duplication charges, time of people involved,
|
||||||
|
and so on. (You will not be required to justify it to the
|
||||||
|
Copyright Holder, but only to the computing community at large
|
||||||
|
as a market that must bear the fee.)
|
||||||
|
|
||||||
|
"Freely Available" means that no fee is charged for the item
|
||||||
|
itself, though there may be fees involved in handling the item.
|
||||||
|
It also means that recipients of the item may redistribute it
|
||||||
|
under the same conditions they received it.
|
||||||
|
|
||||||
|
1. You may make and give away verbatim copies of the source form of the
|
||||||
|
Standard Version of this Package without restriction, provided that you
|
||||||
|
duplicate all of the original copyright notices and associated disclaimers.
|
||||||
|
|
||||||
|
2. You may apply bug fixes, portability fixes and other modifications
|
||||||
|
derived from the Public Domain or from the Copyright Holder. A Package
|
||||||
|
modified in such a way shall still be considered the Standard Version.
|
||||||
|
|
||||||
|
3. You may otherwise modify your copy of this Package in any way, provided
|
||||||
|
that you insert a prominent notice in each changed file stating how and
|
||||||
|
when you changed that file, and provided that you do at least ONE of the
|
||||||
|
following:
|
||||||
|
|
||||||
|
a) place your modifications in the Public Domain or otherwise make them
|
||||||
|
Freely Available, such as by posting said modifications to Usenet or
|
||||||
|
an equivalent medium, or placing the modifications on a major archive
|
||||||
|
site such as uunet.uu.net, or by allowing the Copyright Holder to include
|
||||||
|
your modifications in the Standard Version of the Package.
|
||||||
|
|
||||||
|
b) use the modified Package only within your corporation or organization.
|
||||||
|
|
||||||
|
c) rename any non-standard executables so the names do not conflict
|
||||||
|
with standard executables, which must also be provided, and provide
|
||||||
|
a separate manual page for each non-standard executable that clearly
|
||||||
|
documents how it differs from the Standard Version.
|
||||||
|
|
||||||
|
d) make other distribution arrangements with the Copyright Holder.
|
||||||
|
|
||||||
|
4. You may distribute the programs of this Package in object code or
|
||||||
|
executable form, provided that you do at least ONE of the following:
|
||||||
|
|
||||||
|
a) distribute a Standard Version of the executables and library files,
|
||||||
|
together with instructions (in the manual page or equivalent) on where
|
||||||
|
to get the Standard Version.
|
||||||
|
|
||||||
|
b) accompany the distribution with the machine-readable source of
|
||||||
|
the Package with your modifications.
|
||||||
|
|
||||||
|
c) give non-standard executables non-standard names, and clearly
|
||||||
|
document the differences in manual pages (or equivalent), together
|
||||||
|
with instructions on where to get the Standard Version.
|
||||||
|
|
||||||
|
d) make other distribution arrangements with the Copyright Holder.
|
||||||
|
|
||||||
|
5. You may charge a reasonable copying fee for any distribution of this
|
||||||
|
Package. You may charge any fee you choose for support of this
|
||||||
|
Package. You may not charge a fee for this Package itself. However,
|
||||||
|
you may distribute this Package in aggregate with other (possibly
|
||||||
|
commercial) programs as part of a larger (possibly commercial) software
|
||||||
|
distribution provided that you do not advertise this Package as a
|
||||||
|
product of your own. You may embed this Package's interpreter within
|
||||||
|
an executable of yours (by linking); this shall be construed as a mere
|
||||||
|
form of aggregation, provided that the complete Standard Version of the
|
||||||
|
interpreter is so embedded.
|
||||||
|
|
||||||
|
6. The scripts and library files supplied as input to or produced as
|
||||||
|
output from the programs of this Package do not automatically fall
|
||||||
|
under the copyright of this Package, but belong to whoever generated
|
||||||
|
them, and may be sold commercially, and may be aggregated with this
|
||||||
|
Package. If such scripts or library files are aggregated with this
|
||||||
|
Package via the so-called "undump" or "unexec" methods of producing a
|
||||||
|
binary executable image, then distribution of such an image shall
|
||||||
|
neither be construed as a distribution of this Package nor shall it
|
||||||
|
fall under the restrictions of Paragraphs 3 and 4, provided that you do
|
||||||
|
not represent such an executable image as a Standard Version of this
|
||||||
|
Package.
|
||||||
|
|
||||||
|
7. C subroutines (or comparably compiled subroutines in other
|
||||||
|
languages) supplied by you and linked into this Package in order to
|
||||||
|
emulate subroutines and variables of the language defined by this
|
||||||
|
Package shall not be considered part of this Package, but are the
|
||||||
|
equivalent of input as in Paragraph 6, provided these subroutines do
|
||||||
|
not change the language in any way that would cause it to fail the
|
||||||
|
regression tests for the language.
|
||||||
|
|
||||||
|
8. Aggregation of this Package with a commercial distribution is always
|
||||||
|
permitted provided that the use of this Package is embedded; that is,
|
||||||
|
when no overt attempt is made to make this Package's interfaces visible
|
||||||
|
to the end user of the commercial distribution. Such use shall not be
|
||||||
|
construed as a distribution of this Package.
|
||||||
|
|
||||||
|
9. The name of the Copyright Holder may not be used to endorse or promote
|
||||||
|
products derived from this software without specific prior written permission.
|
||||||
|
|
||||||
|
10. THIS PACKAGE IS PROVIDED "AS IS" AND WITHOUT ANY EXPRESS OR
|
||||||
|
IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
|
||||||
|
WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
|
||||||
|
The End
|
340
additional/Copying
Normal file
340
additional/Copying
Normal file
@ -0,0 +1,340 @@
|
|||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
Version 2, June 1991
|
||||||
|
|
||||||
|
Copyright (C) 1989, 1991 Free Software Foundation, Inc.
|
||||||
|
59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The licenses for most software are designed to take away your
|
||||||
|
freedom to share and change it. By contrast, the GNU General Public
|
||||||
|
License is intended to guarantee your freedom to share and change free
|
||||||
|
software--to make sure the software is free for all its users. This
|
||||||
|
General Public License applies to most of the Free Software
|
||||||
|
Foundation's software and to any other program whose authors commit to
|
||||||
|
using it. (Some other Free Software Foundation software is covered by
|
||||||
|
the GNU Library General Public License instead.) You can apply it to
|
||||||
|
your programs, too.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom, not
|
||||||
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
|
have the freedom to distribute copies of free software (and charge for
|
||||||
|
this service if you wish), that you receive source code or can get it
|
||||||
|
if you want it, that you can change the software or use pieces of it
|
||||||
|
in new free programs; and that you know you can do these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to make restrictions that forbid
|
||||||
|
anyone to deny you these rights or to ask you to surrender the rights.
|
||||||
|
These restrictions translate to certain responsibilities for you if you
|
||||||
|
distribute copies of the software, or if you modify it.
|
||||||
|
|
||||||
|
For example, if you distribute copies of such a program, whether
|
||||||
|
gratis or for a fee, you must give the recipients all the rights that
|
||||||
|
you have. You must make sure that they, too, receive or can get the
|
||||||
|
source code. And you must show them these terms so they know their
|
||||||
|
rights.
|
||||||
|
|
||||||
|
We protect your rights with two steps: (1) copyright the software, and
|
||||||
|
(2) offer you this license which gives you legal permission to copy,
|
||||||
|
distribute and/or modify the software.
|
||||||
|
|
||||||
|
Also, for each author's protection and ours, we want to make certain
|
||||||
|
that everyone understands that there is no warranty for this free
|
||||||
|
software. If the software is modified by someone else and passed on, we
|
||||||
|
want its recipients to know that what they have is not the original, so
|
||||||
|
that any problems introduced by others will not reflect on the original
|
||||||
|
authors' reputations.
|
||||||
|
|
||||||
|
Finally, any free program is threatened constantly by software
|
||||||
|
patents. We wish to avoid the danger that redistributors of a free
|
||||||
|
program will individually obtain patent licenses, in effect making the
|
||||||
|
program proprietary. To prevent this, we have made it clear that any
|
||||||
|
patent must be licensed for everyone's free use or not licensed at all.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow.
|
||||||
|
|
||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||||
|
|
||||||
|
0. This License applies to any program or other work which contains
|
||||||
|
a notice placed by the copyright holder saying it may be distributed
|
||||||
|
under the terms of this General Public License. The "Program", below,
|
||||||
|
refers to any such program or work, and a "work based on the Program"
|
||||||
|
means either the Program or any derivative work under copyright law:
|
||||||
|
that is to say, a work containing the Program or a portion of it,
|
||||||
|
either verbatim or with modifications and/or translated into another
|
||||||
|
language. (Hereinafter, translation is included without limitation in
|
||||||
|
the term "modification".) Each licensee is addressed as "you".
|
||||||
|
|
||||||
|
Activities other than copying, distribution and modification are not
|
||||||
|
covered by this License; they are outside its scope. The act of
|
||||||
|
running the Program is not restricted, and the output from the Program
|
||||||
|
is covered only if its contents constitute a work based on the
|
||||||
|
Program (independent of having been made by running the Program).
|
||||||
|
Whether that is true depends on what the Program does.
|
||||||
|
|
||||||
|
1. You may copy and distribute verbatim copies of the Program's
|
||||||
|
source code as you receive it, in any medium, provided that you
|
||||||
|
conspicuously and appropriately publish on each copy an appropriate
|
||||||
|
copyright notice and disclaimer of warranty; keep intact all the
|
||||||
|
notices that refer to this License and to the absence of any warranty;
|
||||||
|
and give any other recipients of the Program a copy of this License
|
||||||
|
along with the Program.
|
||||||
|
|
||||||
|
You may charge a fee for the physical act of transferring a copy, and
|
||||||
|
you may at your option offer warranty protection in exchange for a fee.
|
||||||
|
|
||||||
|
2. You may modify your copy or copies of the Program or any portion
|
||||||
|
of it, thus forming a work based on the Program, and copy and
|
||||||
|
distribute such modifications or work under the terms of Section 1
|
||||||
|
above, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) You must cause the modified files to carry prominent notices
|
||||||
|
stating that you changed the files and the date of any change.
|
||||||
|
|
||||||
|
b) You must cause any work that you distribute or publish, that in
|
||||||
|
whole or in part contains or is derived from the Program or any
|
||||||
|
part thereof, to be licensed as a whole at no charge to all third
|
||||||
|
parties under the terms of this License.
|
||||||
|
|
||||||
|
c) If the modified program normally reads commands interactively
|
||||||
|
when run, you must cause it, when started running for such
|
||||||
|
interactive use in the most ordinary way, to print or display an
|
||||||
|
announcement including an appropriate copyright notice and a
|
||||||
|
notice that there is no warranty (or else, saying that you provide
|
||||||
|
a warranty) and that users may redistribute the program under
|
||||||
|
these conditions, and telling the user how to view a copy of this
|
||||||
|
License. (Exception: if the Program itself is interactive but
|
||||||
|
does not normally print such an announcement, your work based on
|
||||||
|
the Program is not required to print an announcement.)
|
||||||
|
|
||||||
|
These requirements apply to the modified work as a whole. If
|
||||||
|
identifiable sections of that work are not derived from the Program,
|
||||||
|
and can be reasonably considered independent and separate works in
|
||||||
|
themselves, then this License, and its terms, do not apply to those
|
||||||
|
sections when you distribute them as separate works. But when you
|
||||||
|
distribute the same sections as part of a whole which is a work based
|
||||||
|
on the Program, the distribution of the whole must be on the terms of
|
||||||
|
this License, whose permissions for other licensees extend to the
|
||||||
|
entire whole, and thus to each and every part regardless of who wrote it.
|
||||||
|
|
||||||
|
Thus, it is not the intent of this section to claim rights or contest
|
||||||
|
your rights to work written entirely by you; rather, the intent is to
|
||||||
|
exercise the right to control the distribution of derivative or
|
||||||
|
collective works based on the Program.
|
||||||
|
|
||||||
|
In addition, mere aggregation of another work not based on the Program
|
||||||
|
with the Program (or with a work based on the Program) on a volume of
|
||||||
|
a storage or distribution medium does not bring the other work under
|
||||||
|
the scope of this License.
|
||||||
|
|
||||||
|
3. You may copy and distribute the Program (or a work based on it,
|
||||||
|
under Section 2) in object code or executable form under the terms of
|
||||||
|
Sections 1 and 2 above provided that you also do one of the following:
|
||||||
|
|
||||||
|
a) Accompany it with the complete corresponding machine-readable
|
||||||
|
source code, which must be distributed under the terms of Sections
|
||||||
|
1 and 2 above on a medium customarily used for software interchange; or,
|
||||||
|
|
||||||
|
b) Accompany it with a written offer, valid for at least three
|
||||||
|
years, to give any third party, for a charge no more than your
|
||||||
|
cost of physically performing source distribution, a complete
|
||||||
|
machine-readable copy of the corresponding source code, to be
|
||||||
|
distributed under the terms of Sections 1 and 2 above on a medium
|
||||||
|
customarily used for software interchange; or,
|
||||||
|
|
||||||
|
c) Accompany it with the information you received as to the offer
|
||||||
|
to distribute corresponding source code. (This alternative is
|
||||||
|
allowed only for noncommercial distribution and only if you
|
||||||
|
received the program in object code or executable form with such
|
||||||
|
an offer, in accord with Subsection b above.)
|
||||||
|
|
||||||
|
The source code for a work means the preferred form of the work for
|
||||||
|
making modifications to it. For an executable work, complete source
|
||||||
|
code means all the source code for all modules it contains, plus any
|
||||||
|
associated interface definition files, plus the scripts used to
|
||||||
|
control compilation and installation of the executable. However, as a
|
||||||
|
special exception, the source code distributed need not include
|
||||||
|
anything that is normally distributed (in either source or binary
|
||||||
|
form) with the major components (compiler, kernel, and so on) of the
|
||||||
|
operating system on which the executable runs, unless that component
|
||||||
|
itself accompanies the executable.
|
||||||
|
|
||||||
|
If distribution of executable or object code is made by offering
|
||||||
|
access to copy from a designated place, then offering equivalent
|
||||||
|
access to copy the source code from the same place counts as
|
||||||
|
distribution of the source code, even though third parties are not
|
||||||
|
compelled to copy the source along with the object code.
|
||||||
|
|
||||||
|
4. You may not copy, modify, sublicense, or distribute the Program
|
||||||
|
except as expressly provided under this License. Any attempt
|
||||||
|
otherwise to copy, modify, sublicense or distribute the Program is
|
||||||
|
void, and will automatically terminate your rights under this License.
|
||||||
|
However, parties who have received copies, or rights, from you under
|
||||||
|
this License will not have their licenses terminated so long as such
|
||||||
|
parties remain in full compliance.
|
||||||
|
|
||||||
|
5. You are not required to accept this License, since you have not
|
||||||
|
signed it. However, nothing else grants you permission to modify or
|
||||||
|
distribute the Program or its derivative works. These actions are
|
||||||
|
prohibited by law if you do not accept this License. Therefore, by
|
||||||
|
modifying or distributing the Program (or any work based on the
|
||||||
|
Program), you indicate your acceptance of this License to do so, and
|
||||||
|
all its terms and conditions for copying, distributing or modifying
|
||||||
|
the Program or works based on it.
|
||||||
|
|
||||||
|
6. Each time you redistribute the Program (or any work based on the
|
||||||
|
Program), the recipient automatically receives a license from the
|
||||||
|
original licensor to copy, distribute or modify the Program subject to
|
||||||
|
these terms and conditions. You may not impose any further
|
||||||
|
restrictions on the recipients' exercise of the rights granted herein.
|
||||||
|
You are not responsible for enforcing compliance by third parties to
|
||||||
|
this License.
|
||||||
|
|
||||||
|
7. If, as a consequence of a court judgment or allegation of patent
|
||||||
|
infringement or for any other reason (not limited to patent issues),
|
||||||
|
conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot
|
||||||
|
distribute so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you
|
||||||
|
may not distribute the Program at all. For example, if a patent
|
||||||
|
license would not permit royalty-free redistribution of the Program by
|
||||||
|
all those who receive copies directly or indirectly through you, then
|
||||||
|
the only way you could satisfy both it and this License would be to
|
||||||
|
refrain entirely from distribution of the Program.
|
||||||
|
|
||||||
|
If any portion of this section is held invalid or unenforceable under
|
||||||
|
any particular circumstance, the balance of the section is intended to
|
||||||
|
apply and the section as a whole is intended to apply in other
|
||||||
|
circumstances.
|
||||||
|
|
||||||
|
It is not the purpose of this section to induce you to infringe any
|
||||||
|
patents or other property right claims or to contest validity of any
|
||||||
|
such claims; this section has the sole purpose of protecting the
|
||||||
|
integrity of the free software distribution system, which is
|
||||||
|
implemented by public license practices. Many people have made
|
||||||
|
generous contributions to the wide range of software distributed
|
||||||
|
through that system in reliance on consistent application of that
|
||||||
|
system; it is up to the author/donor to decide if he or she is willing
|
||||||
|
to distribute software through any other system and a licensee cannot
|
||||||
|
impose that choice.
|
||||||
|
|
||||||
|
This section is intended to make thoroughly clear what is believed to
|
||||||
|
be a consequence of the rest of this License.
|
||||||
|
|
||||||
|
8. If the distribution and/or use of the Program is restricted in
|
||||||
|
certain countries either by patents or by copyrighted interfaces, the
|
||||||
|
original copyright holder who places the Program under this License
|
||||||
|
may add an explicit geographical distribution limitation excluding
|
||||||
|
those countries, so that distribution is permitted only in or among
|
||||||
|
countries not thus excluded. In such case, this License incorporates
|
||||||
|
the limitation as if written in the body of this License.
|
||||||
|
|
||||||
|
9. The Free Software Foundation may publish revised and/or new versions
|
||||||
|
of the General Public License from time to time. Such new versions will
|
||||||
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
|
address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the Program
|
||||||
|
specifies a version number of this License which applies to it and "any
|
||||||
|
later version", you have the option of following the terms and conditions
|
||||||
|
either of that version or of any later version published by the Free
|
||||||
|
Software Foundation. If the Program does not specify a version number of
|
||||||
|
this License, you may choose any version ever published by the Free Software
|
||||||
|
Foundation.
|
||||||
|
|
||||||
|
10. If you wish to incorporate parts of the Program into other free
|
||||||
|
programs whose distribution conditions are different, write to the author
|
||||||
|
to ask for permission. For software which is copyrighted by the Free
|
||||||
|
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||||
|
make exceptions for this. Our decision will be guided by the two goals
|
||||||
|
of preserving the free status of all derivatives of our free software and
|
||||||
|
of promoting the sharing and reuse of software generally.
|
||||||
|
|
||||||
|
NO WARRANTY
|
||||||
|
|
||||||
|
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||||
|
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||||
|
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||||
|
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||||
|
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
|
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||||
|
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||||
|
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||||
|
REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||||
|
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||||
|
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||||
|
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||||
|
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||||
|
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||||
|
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||||
|
POSSIBILITY OF SUCH DAMAGES.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Programs
|
||||||
|
|
||||||
|
If you develop a new program, and you want it to be of the greatest
|
||||||
|
possible use to the public, the best way to achieve this is to make it
|
||||||
|
free software which everyone can redistribute and change under these terms.
|
||||||
|
|
||||||
|
To do so, attach the following notices to the program. It is safest
|
||||||
|
to attach them to the start of each source file to most effectively
|
||||||
|
convey the exclusion of warranty; and each file should have at least
|
||||||
|
the "copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the program's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) 19yy <name of author>
|
||||||
|
|
||||||
|
This program is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation; either version 2 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
If the program is interactive, make it output a short notice like this
|
||||||
|
when it starts in an interactive mode:
|
||||||
|
|
||||||
|
Gnomovision version 69, Copyright (C) 19yy name of author
|
||||||
|
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||||
|
This is free software, and you are welcome to redistribute it
|
||||||
|
under certain conditions; type `show c' for details.
|
||||||
|
|
||||||
|
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||||
|
parts of the General Public License. Of course, the commands you use may
|
||||||
|
be called something other than `show w' and `show c'; they could even be
|
||||||
|
mouse-clicks or menu items--whatever suits your program.
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or your
|
||||||
|
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||||
|
necessary. Here is a sample; alter the names:
|
||||||
|
|
||||||
|
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||||
|
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||||
|
|
||||||
|
<signature of Ty Coon>, 1 April 1989
|
||||||
|
Ty Coon, President of Vice
|
||||||
|
|
||||||
|
This General Public License does not permit incorporating your program into
|
||||||
|
proprietary programs. If your program is a subroutine library, you may
|
||||||
|
consider it more useful to permit linking proprietary applications with the
|
||||||
|
library. If this is what you want to do, use the GNU Library General
|
||||||
|
Public License instead of this License.
|
33
additional/LICENSE
Normal file
33
additional/LICENSE
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
|
||||||
|
|
||||||
|
Perl Kit, Version 5.0
|
||||||
|
|
||||||
|
Copyright 1989-2001, Larry Wall
|
||||||
|
All rights reserved.
|
||||||
|
|
||||||
|
This program is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of either:
|
||||||
|
|
||||||
|
a) the GNU General Public License as published by the Free
|
||||||
|
Software Foundation; either version 1, or (at your option) any
|
||||||
|
later version, or
|
||||||
|
|
||||||
|
b) the "Artistic License" which comes with this Kit.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See either
|
||||||
|
the GNU General Public License or the Artistic License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the Artistic License with this
|
||||||
|
Kit, in the file named "Artistic". If not, I'll be glad to provide one.
|
||||||
|
|
||||||
|
You should also have received a copy of the GNU General Public License
|
||||||
|
along with this program in the file named "Copying". If not, write to the
|
||||||
|
Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
|
||||||
|
02111-1307, USA or visit their web page on the internet at
|
||||||
|
http://www.gnu.org/copyleft/gpl.html.
|
||||||
|
|
790
additional/e-smith-lib.spec
Normal file
790
additional/e-smith-lib.spec
Normal file
@ -0,0 +1,790 @@
|
|||||||
|
# $Id: e-smith-lib.spec,v 1.24 2010/10/11 22:27:51 slords Exp $
|
||||||
|
|
||||||
|
Summary: e-smith server and gateway - library module
|
||||||
|
%define name e-smith-lib
|
||||||
|
Name: %{name}
|
||||||
|
%define version 2.4.0
|
||||||
|
%define release 1
|
||||||
|
Version: %{version}
|
||||||
|
Release: %{release}%{?dist}
|
||||||
|
License: Artistic
|
||||||
|
Group: Networking/Daemons
|
||||||
|
Source: %{name}-%{version}.tar.xz
|
||||||
|
BuildRoot: /var/tmp/%{name}-%{version}-%{release}-buildroot
|
||||||
|
BuildArchitectures: noarch
|
||||||
|
BuildRequires: e-smith-devtools >= 1.6.3-01
|
||||||
|
Obsoletes: %{name}-Tai64n
|
||||||
|
Obsoletes: whiptail
|
||||||
|
Requires: dialog
|
||||||
|
Requires: perl, perl(Text::Template)
|
||||||
|
Requires: perl(Time::HiRes), perl(MIME::Base64)
|
||||||
|
Requires: perl(Authen::PAM), perl(I18N::AcceptLanguage)
|
||||||
|
Requires: perl(I18N::LangTags) >= 0.27
|
||||||
|
Requires: perl(Net::IPv4Addr) >= 0.10
|
||||||
|
|
||||||
|
%description
|
||||||
|
e-smith server and gateway software - library module.
|
||||||
|
|
||||||
|
%changelog
|
||||||
|
* Fri Jan 25 2013 Shad L. Lords <slords@mail.com> 2.4.0-1.sme
|
||||||
|
- Bump version in prep for SME9
|
||||||
|
|
||||||
|
* Mon Oct 11 2010 Shad L. Lords <slords@mail.com> 2.2.0-7.sme
|
||||||
|
- Serialize configure_peers to prevent errors [SME: 5831]
|
||||||
|
|
||||||
|
* Fri Feb 5 2010 Stephen Noble <support@dungog.net> 2.2.0-6.sme
|
||||||
|
- adds the hwaddr parameter to probeAdapters() [SME: 4528]
|
||||||
|
|
||||||
|
* Thu Feb 4 2010 Shad L. Lords <slords@mail.com> 2.2.0-5.sme
|
||||||
|
- Update path for 64-bit compatibility [SME: 5756]
|
||||||
|
|
||||||
|
* Tue Dec 22 2009 Filippo Carletti <filippo.carletti@gmail.com> 2.2.0-4.sme
|
||||||
|
- Really apply previous patch in the spec file. [SME: 5659]
|
||||||
|
|
||||||
|
* Wed Dec 9 2009 Charlie Brady <charlieb@budge.apana.org.au> 2.2.0-3.sme
|
||||||
|
- Add patch (Federico Simoncelli) to prevent re-use of uids. [SME: 5659]
|
||||||
|
|
||||||
|
* Mon Oct 13 2008 Shad L. Lords <slords@mail.com> 2.2.0-2.sme
|
||||||
|
- Add patch to support multiple samba roles [SME: 4172]
|
||||||
|
|
||||||
|
* Tue Oct 7 2008 Shad L. Lords <slords@mail.com> 2.2.0-1.sme
|
||||||
|
- Roll new stream to separate sme7/sme8 trees [SME: 4633]
|
||||||
|
|
||||||
|
* Wed Aug 20 2008 Shad L. Lords <slords@mail.com> 1.19.0-1
|
||||||
|
- Roll new dev stream.
|
||||||
|
|
||||||
|
* Sat Aug 9 2008 Shad L. Lords <slords@mail.com> 1.18.0-28
|
||||||
|
- Read /dev/urandmon instead of /dev/random [SME: 4492]
|
||||||
|
|
||||||
|
* Sat Aug 9 2008 Shad L. Lords <slords@mail.com> 1.18.0-27
|
||||||
|
- Make dialog not use STDERR for return values [SME: 3663]
|
||||||
|
- Remove requires for whiptail. No longer needed [SME: 4491]
|
||||||
|
|
||||||
|
* Thu Jul 31 2008 Shad L. Lords <slords@mail.com> 1.18.0-26
|
||||||
|
- Make binmode properties of db class [SME: 4317]
|
||||||
|
- Add new navigation db & utf8 classes [SME: 4317]
|
||||||
|
|
||||||
|
* Wed Apr 2 2008 Shad L. Lords <slords@mail.com> 1.18.0-25
|
||||||
|
- Add .utf8 to locale unless server-console [SME: 4162]
|
||||||
|
- Add new languages to langtag2locale [SME: 4163]
|
||||||
|
|
||||||
|
* Sun Mar 30 2008 Shad L. Lords <slords@mail.com> 1.18.0-24
|
||||||
|
- Gracefully handle encoding errors in navigation dbs [SME: 4147]
|
||||||
|
|
||||||
|
* Wed Mar 26 2008 Shad L. Lords <slords@mail.com> 1.18.0-23
|
||||||
|
- Make PAM conv routine not look for english strings [SME: 4117]
|
||||||
|
|
||||||
|
* Tue Mar 25 2008 Shad L. Lords <slords@mail.com> 1.18.0-22
|
||||||
|
- Fix wide output to syslog and allow navigations db to be utf8 [SME: 4101]
|
||||||
|
|
||||||
|
* Mon Mar 10 2008 Shad L. Lords <slords@mail.com> 1.18.0-21
|
||||||
|
- Output navigation in UTF-8 [SME: 3858]
|
||||||
|
|
||||||
|
* Sun Jan 20 2008 Gavin Weight <gweight@gmail.com> 1.18.0-20
|
||||||
|
- Call smbpasswd -e - to reset user correctly. [SME: 3755]
|
||||||
|
|
||||||
|
* Tue Jan 08 2008 Stephen Noble <support@dungog.net> 1.18.0-19
|
||||||
|
- modify validate password strong match for console [SME: 2173]
|
||||||
|
|
||||||
|
* Thu Nov 01 2007 Gavin Weight <gweight@gmail.com> 1.18.0-18
|
||||||
|
- Fix I18N.pm file descriptor leak. [SME: 3509]
|
||||||
|
|
||||||
|
* Wed Oct 31 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-17
|
||||||
|
- Remove undocumented and unused template metadata handling from
|
||||||
|
generic_template_expand action (Take II). [SME: 2798]
|
||||||
|
|
||||||
|
* Fri Jun 1 2007 Shad L. Lords <slords@mail.com> 1.18.0-16
|
||||||
|
- pam_unix requires passwords >= 6 [SME: 3039]
|
||||||
|
|
||||||
|
* Mon May 21 2007 Shad L. Lords <slords@mail.com> 1.18.0-15
|
||||||
|
- Fix ip addr shift to work on 64-bit
|
||||||
|
|
||||||
|
* Sun Apr 29 2007 Shad L. Lords <slords@mail.com>
|
||||||
|
- Clean up spec so package can be built by koji/plague
|
||||||
|
|
||||||
|
* Thu Apr 05 2007 Shad L. Lords <slords@mail.com> 1.18.0-14
|
||||||
|
- Reverse changes for 1.18.0-12 as it is being used. [SME: 2838]
|
||||||
|
|
||||||
|
* Wed Apr 04 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-13
|
||||||
|
- Allow gauge console widget to be used without following message_page.
|
||||||
|
[SME: 2832].
|
||||||
|
|
||||||
|
* Mon Mar 26 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-12
|
||||||
|
- Remove undocumented and unused template metadata handling from
|
||||||
|
generic_template_expand action. [SME: 2798]
|
||||||
|
|
||||||
|
* Mon Mar 26 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-11
|
||||||
|
- Split logfile rotation code out of generic_template_expand.
|
||||||
|
Code moved to a new action in e-smith-base. [SME: 2795]
|
||||||
|
|
||||||
|
* Mon Mar 19 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-10
|
||||||
|
Don't try to read template.metadata from subdirectories (affects
|
||||||
|
initialize_database_defaults). [SME: 2690]
|
||||||
|
|
||||||
|
* Fri Mar 16 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-9
|
||||||
|
- Allow template.metadata to be provided via a directory of
|
||||||
|
small files. [SME: 2690]
|
||||||
|
- Add support for a DELETE boolean template.metadata item. [SME: 2691]
|
||||||
|
|
||||||
|
* Wed Feb 28 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-8
|
||||||
|
- Add gauge widget to esmith::console. [SME: 2579]
|
||||||
|
|
||||||
|
* Sun Feb 25 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-7
|
||||||
|
- Default infobox height to 8 rows, and allow override via params.
|
||||||
|
[SME: 2560]
|
||||||
|
|
||||||
|
* Fri Feb 23 2007 Shad L. Lords <slords@mail.com> 1.18.0-6
|
||||||
|
- Really fix clear parameter for inputbox dialog screen [SME: 2533]
|
||||||
|
|
||||||
|
* Fri Feb 23 2007 Shad L. Lords <slords@mail.com> 1.18.0-5
|
||||||
|
- Fix clear parameter for dialog screens [SME: 2533]
|
||||||
|
|
||||||
|
* Thu Feb 22 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-4
|
||||||
|
- switch esmith::console::password_page() to use dialog rather than
|
||||||
|
whiptail. [SME: 2534]
|
||||||
|
|
||||||
|
* Thu Feb 22 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-3
|
||||||
|
- Add infobox widget. [SME: 2533]
|
||||||
|
|
||||||
|
* Sun Jan 28 2007 Shad L. Lords <slords@mail.com> 1.18.0-2
|
||||||
|
- Fix backtitle spacing for new dialog menus [SME: 2328]
|
||||||
|
|
||||||
|
* Fri Jan 26 2007 Shad L. Lords <slords@mail.com> 1.18.0-1
|
||||||
|
- Roll stable stream. [SME: 2328]
|
||||||
|
|
||||||
|
* Tue Dec 15 2006 Federico Simoncelli <federico.simoncelli@gmail.com> 1.17.0-8
|
||||||
|
- Added the validatePassword function to esmith::util. [SME: 2100]
|
||||||
|
|
||||||
|
* Thu Dec 07 2006 Shad L. Lords <slords@mail.com>
|
||||||
|
- Update to new release naming. No functional changes.
|
||||||
|
- Make Packager generic
|
||||||
|
|
||||||
|
* Fri Nov 24 2006 Gordon Rowell <gordonr@gormand.com.au> 1.17.0-06
|
||||||
|
- Allow optional path to peeers directory in
|
||||||
|
esmith::tcpsvd::configure_peers() [SME: 2086]
|
||||||
|
|
||||||
|
* Tue Nov 14 2006 Charlie Brady <charlie_brady@mitel.com> 1.17.0-05
|
||||||
|
- Allow defaulno option to esmith::console::yesno.
|
||||||
|
- Fix I/O redirection in esmith::console::new.
|
||||||
|
|
||||||
|
* Tue Nov 07 2006 Charlie Brady <charlie_brady@mitel.com> 1.17.0-04
|
||||||
|
- Allow display of "0" in iesmith::cgi::genSmallCell. [SME: 2038]
|
||||||
|
|
||||||
|
* Mon Oct 23 2006 Charlie Brady <charlie_brady@mitel.com> 1.17.0-03
|
||||||
|
- Make dialog the default console app, with whiptail used only when
|
||||||
|
required.
|
||||||
|
|
||||||
|
* Wed Oct 11 2006 Charlie Brady <charlie_brady@mitel.com> 1.17.0-02
|
||||||
|
- Allow 'dialog' to be called from esmith::console. [SME: 1958]
|
||||||
|
|
||||||
|
* Wed Oct 11 2006 Charlie Brady <charlie_brady@mitel.com> 1.17.0-01
|
||||||
|
- Roll development branch.
|
||||||
|
|
||||||
|
* Mon Sep 25 2006 Charlie Brady <charlie_brady@mitel.com> 1.16.0-06
|
||||||
|
- Fix problem with greedy RE in template.metadata parsing. [SME: 1906]
|
||||||
|
|
||||||
|
* Fri Sep 08 2006 Charlie Brady <charlie_brady@mitel.com> 1.16.0-05
|
||||||
|
- Fix taint problem in template.metadata handling. [SME: 1906]
|
||||||
|
|
||||||
|
* Thu Apr 13 2006 Charlie Brady <charlie_brady@mitel.com> 1.16.0-04
|
||||||
|
- Use "kudzu --probe --class network" for NIC detection. [SME: 727]
|
||||||
|
|
||||||
|
* Fri Apr 7 2006 Gordon Rowell <gordonr@gormand.com.au> 1.16.0-03
|
||||||
|
- Don't call smbpasswd -e - setting the password is sufficient [SME: 1193]
|
||||||
|
|
||||||
|
* Tue Mar 28 2006 Gordon Rowell <gordonr@gormand.com.au> 1.16.0-02
|
||||||
|
- Log previous contents of db entry in DELETE log [SME: 1066]
|
||||||
|
|
||||||
|
* Tue Mar 14 2006 Charlie Brady <charlie_brady@mitel.com> 1.16.0-01
|
||||||
|
- Roll to stable stream version number. [SME: 1016]
|
||||||
|
|
||||||
|
* Fri Mar 10 2006 Charlie Brady <charlie_brady@mitel.com> 1.15.4-02
|
||||||
|
- Suppress warning from genSmallCell if text is undefined. [SME: 986]
|
||||||
|
|
||||||
|
* Fri Feb 17 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.4-01
|
||||||
|
- Roll patches up to 1.15.3-42
|
||||||
|
- Trim changelog prior to 1.15.0-01 [SME: 828]
|
||||||
|
|
||||||
|
* Thu Feb 16 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-42
|
||||||
|
- Adjust console title bar to 'SME Server' [SME: 726]
|
||||||
|
|
||||||
|
* Tue Feb 14 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-41
|
||||||
|
- Reworded text for template-begin and change URL to /development/
|
||||||
|
rather than /custom/ [SME: 773]
|
||||||
|
|
||||||
|
* Mon Feb 13 2006 Charlie Brady <charlie_brady@mitel.com> 1.15.3-40
|
||||||
|
- Update URL in default template-begin fragment. [SME: 773]
|
||||||
|
|
||||||
|
* Sat Feb 11 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-39
|
||||||
|
- Remove obsolete e-smith-lib-Tai64n package. [SME: 689]
|
||||||
|
|
||||||
|
* Sat Feb 11 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-38
|
||||||
|
- [Null changelog for missing version - we accidentally skipped
|
||||||
|
this version.]
|
||||||
|
|
||||||
|
* Sat Feb 11 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-37
|
||||||
|
- Fix get_all_by_prop in scalar context. [SME: 669,721]
|
||||||
|
|
||||||
|
* Mon Feb 6 2006 Shad L. Lords <slords@mail.com> 1.15.3-37
|
||||||
|
- Add ability to pass many props to get_all_by_prop [SME: 669]
|
||||||
|
|
||||||
|
* Mon Jan 23 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-36
|
||||||
|
- Fix warning during pseudonym deletion. [SME: 491]
|
||||||
|
|
||||||
|
* Fri Jan 20 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-35
|
||||||
|
- Fix up use of Sys::Syslog::syslog. [SME: 526]
|
||||||
|
|
||||||
|
* Thu Jan 19 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-34
|
||||||
|
- Remove running of kudzu during NIC probing. TODO: Find a solution
|
||||||
|
to system reconfiguration when new hardware is added. [SME: 192]
|
||||||
|
|
||||||
|
* Tue Jan 10 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-33
|
||||||
|
- Fold a.b.c.d/255.255.255.255 to a.b.c.d in local_access_spec() to
|
||||||
|
work around bugs in applications which don't accept such specs.
|
||||||
|
[SME: 430]
|
||||||
|
|
||||||
|
* Mon Jan 9 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-32
|
||||||
|
- Check whether an accounts db record exists before trying to create
|
||||||
|
the dot and underscore pseudonyms (new_record will fail silently)
|
||||||
|
and check that the records are pseudonyms before deleting them [SME: 24]
|
||||||
|
|
||||||
|
* Mon Jan 9 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-31
|
||||||
|
- And update POD for last change [SME: 24]
|
||||||
|
|
||||||
|
* Mon Jan 9 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-30
|
||||||
|
- Allow dot and underscore in account names [SME: 24]
|
||||||
|
|
||||||
|
* Tue Dec 27 2005 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-29
|
||||||
|
- Remove newlines from cluck() string and just note that the old
|
||||||
|
path was used [SME: 365]
|
||||||
|
|
||||||
|
* Sat Dec 25 2005 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-28
|
||||||
|
- If db exists in both the old and new locations in
|
||||||
|
initialize_default_databases, rename the one in the new
|
||||||
|
location to db.time(), avoiding the conflict and saving the
|
||||||
|
evidence in case it is needed later [SME: 229]
|
||||||
|
|
||||||
|
* Mon Dec 05 2005 Mark Knox <mark_knox@mitel.com>
|
||||||
|
- [1.15.3-27]
|
||||||
|
- Moved queueing logic to separate daemon, replaced with FIFO IPC [BZ252
|
||||||
|
|
||||||
|
* Thu Dec 01 2005 Mark Knox <mark_knox@mitel.com>
|
||||||
|
- [1.15.3-26]
|
||||||
|
- Added event queueing (open source portion) for clustered systems [BZ250]
|
||||||
|
|
||||||
|
* Wed Nov 30 2005 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-25
|
||||||
|
- Bump release number only
|
||||||
|
|
||||||
|
* Thu Nov 24 2005 Gordon Rowell <gordonr@e-smith.com>
|
||||||
|
- [1.15.3-24]
|
||||||
|
- Add missing 'use Locale::gettext' to esmith::console.pm [MN00108804]
|
||||||
|
|
||||||
|
* Sun Nov 20 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-23]
|
||||||
|
- Clarify logic for stopped services in adjust-services. [SF: 1357629]
|
||||||
|
|
||||||
|
* Sun Nov 20 2005 Gordon Rowell <gordonr@e-smith.com>
|
||||||
|
- [1.15.3-22]
|
||||||
|
- Correct adjust-services logic for stopped services [SF: 1357629]
|
||||||
|
|
||||||
|
* Wed Nov 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-21]
|
||||||
|
- Allow services2adjust directories to contain files rather than just dangling
|
||||||
|
symlinks. Files can contain more than one actions to perform. [SF: 1270644]
|
||||||
|
|
||||||
|
* Wed Nov 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-20]
|
||||||
|
- Also don't start services if we just want to "once" them. [SF: 1357629]
|
||||||
|
|
||||||
|
* Wed Nov 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-19]
|
||||||
|
- Fix restart of enabled supervised services which we are attempting to stop.
|
||||||
|
[SF: 1357629]
|
||||||
|
|
||||||
|
* Tue Nov 15 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-18]
|
||||||
|
- Set default for $type in esmith::cgi::genSmallCell, to prevent some log
|
||||||
|
noise. [SF: 1357830]
|
||||||
|
|
||||||
|
* Tue Nov 15 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-17]
|
||||||
|
- Pass $EVENT to template expansions in generic_template_expand.
|
||||||
|
[SF: MN00106104]
|
||||||
|
|
||||||
|
* Tue Nov 15 2005 Gordon Rowell <gordonr@e-smith.com>
|
||||||
|
- [1.15.3-16]
|
||||||
|
- Redirect esmith::config calls on old db paths to the new
|
||||||
|
location [SF: 1335865]
|
||||||
|
|
||||||
|
* Thu Oct 27 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-15]
|
||||||
|
- Fix a few minor spec file portability issues. [SF: 1339729]
|
||||||
|
|
||||||
|
* Wed Oct 26 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-14]
|
||||||
|
- Add it and de to the langtag2locale fixups. [SF: 1338236]
|
||||||
|
|
||||||
|
* Tue Oct 11 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-13]
|
||||||
|
- Build driver list from .ko files as well as .o files. Untaint driver
|
||||||
|
names while building list. [SF: 1323270]
|
||||||
|
|
||||||
|
* Mon Sep 26 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-12]
|
||||||
|
- Fix "defaults" handling so that values which evaluate to false are
|
||||||
|
not overridden by default. [SF: 1303885]
|
||||||
|
|
||||||
|
* Fri Sep 23 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-11]
|
||||||
|
- Untaint db names before attempting to move them. [MN00098405]
|
||||||
|
|
||||||
|
* Thu Sep 22 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-10]
|
||||||
|
- Provide networks method in esmith::NetworksDB. [SF: 1296099]
|
||||||
|
- Support a "localhost" configuration in esmith::tcpsvd:configure_peers
|
||||||
|
[SF: 1294719]
|
||||||
|
|
||||||
|
* Tue Sep 20 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-09]
|
||||||
|
- Remove deprecated functions from esmith::utils. [SF: 1295851]
|
||||||
|
- Include only "network" records in local_access_spec. [SF: 1296099]
|
||||||
|
|
||||||
|
* Mon Sep 12 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-08]
|
||||||
|
- Remove warning about explicit path in esmith::db::_db_path.
|
||||||
|
[SF: 1286294]
|
||||||
|
|
||||||
|
* Fri Sep 9 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-07]
|
||||||
|
- Tolerate, but warn about, symlinks in /home/e-smith. [SF: 1216546]
|
||||||
|
|
||||||
|
* Fri Sep 9 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-06]
|
||||||
|
- Reduce the noise from _file_path() in esmith::DB::db. [SF: 1286294]
|
||||||
|
|
||||||
|
* Wed Sep 7 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-05]
|
||||||
|
- Fix operation of expandTemplate when taint check is enabled.
|
||||||
|
[SF: 1284301]
|
||||||
|
|
||||||
|
* Wed Aug 17 2005 Mark Knox <markk@e-smith.com>
|
||||||
|
- [1.15.3-04]
|
||||||
|
- Added open_local and open_ro_local methods for clustering [markk MN00094831]
|
||||||
|
|
||||||
|
* Tue Aug 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-03]
|
||||||
|
- Fix POD error in util.pm.
|
||||||
|
|
||||||
|
* Wed Jul 27 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-02]
|
||||||
|
- Move databases default location from /home/e-smith to /home/e-smith/db.
|
||||||
|
During esmith::utils::initialize_default_databases, move from old to new
|
||||||
|
location before doing db migrate actions. [SF: 1216546]
|
||||||
|
|
||||||
|
* Wed Jul 27 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-01]
|
||||||
|
- Roll a new development stream - 1.15.3
|
||||||
|
|
||||||
|
* Wed Jul 27 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.2-04]
|
||||||
|
- Remove broken MergeDB stuff. [SF: 1246315]
|
||||||
|
|
||||||
|
* Wed Jul 27 2005 Mark Knox <markk@e-smith.com>
|
||||||
|
- [1.15.2-03]
|
||||||
|
- Fixed misleading comment in util.pm
|
||||||
|
- Added 'use' statements in Record classes for SOAP compatibility
|
||||||
|
- Fixed undefined max_len warning
|
||||||
|
- Added explicit writeconf calls in db::Record.pm, needed for setting props
|
||||||
|
via SOAP
|
||||||
|
|
||||||
|
* Tue Jul 19 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.2-02]
|
||||||
|
- Allow db open API to use default path if a simple filename
|
||||||
|
is provided as arg. This is preparation for move of dbs to
|
||||||
|
/home/e-smith/db. Patch by Shad Lords.
|
||||||
|
|
||||||
|
* Mon Jul 18 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.2-01]
|
||||||
|
- Roll new development stream - 1.15.2
|
||||||
|
|
||||||
|
* Fri Jul 15 2005 Mark Knox <markk@e-smith.com>
|
||||||
|
- [1.15.1-44]
|
||||||
|
- Tweak to allow calling _writeconf from SOAP [markk MN00090738]
|
||||||
|
|
||||||
|
* Tue Jun 21 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-43]
|
||||||
|
- Ensure that esmith::util::LdapPassword returns bare string without
|
||||||
|
newline terminator.
|
||||||
|
|
||||||
|
* Sun Jun 12 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-42]
|
||||||
|
- Remove .rpmsave and .rpmnew symlinks (as well as files). [SF: 1217969]
|
||||||
|
- Handle missing description in pcitables entries.
|
||||||
|
|
||||||
|
* Sun Jun 12 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-41]
|
||||||
|
- Provide feedback (via log messages) from services2adjust. [SF: 1218920]
|
||||||
|
|
||||||
|
* Mon May 30 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-40]
|
||||||
|
- Add preinstall scripting to create required accounts/groups if they
|
||||||
|
don't already exist. [SF: 1210723]
|
||||||
|
|
||||||
|
* Thu May 5 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-39]
|
||||||
|
- Show Text::Template error text rather than inappropriate $! if template
|
||||||
|
expansion fails.
|
||||||
|
- Change error to warning if a config item is set with an empty "type"
|
||||||
|
property.
|
||||||
|
|
||||||
|
* Thu May 5 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-38]
|
||||||
|
- Fix esmith::DB::get_prop_and_delete fail if prop is "false" [From Gordon].
|
||||||
|
|
||||||
|
* Tue May 3 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-37]
|
||||||
|
- Update directory list so that ethernet drivers from kernel-unsupported are
|
||||||
|
added to "choose by driver" list.
|
||||||
|
|
||||||
|
* Sat Mar 19 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-36]
|
||||||
|
- Rework esmith::tcpsvd::configure_peers so that it doesn't chdir.
|
||||||
|
- Fix generic_template_expand so that templates.metadata doesn't
|
||||||
|
need to set OUTPUT_FILENAME is TEMPLATE_PATH is changed. This
|
||||||
|
matches what expand-template already does.
|
||||||
|
|
||||||
|
* Fri Mar 18 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-35]
|
||||||
|
- Change numerous calls to "croak" in esmith::template
|
||||||
|
to "carp ...; return", so that problem with any single
|
||||||
|
template expansion doesn't terminate calling program.
|
||||||
|
[MN00075009]
|
||||||
|
|
||||||
|
* Wed Mar 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-34]
|
||||||
|
- Add missing "use esmith::util" in esmith::tcpsvd.
|
||||||
|
|
||||||
|
* Wed Mar 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-33]
|
||||||
|
- Add esmith::tcpsvd library for managing tcpsvd "peers"
|
||||||
|
directories.
|
||||||
|
|
||||||
|
* Mon Mar 14 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-32]
|
||||||
|
- Make template expansion message more succinct.
|
||||||
|
|
||||||
|
* Thu Mar 10 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-31]
|
||||||
|
- Remove pseudonyms of pseudonyms when removing user
|
||||||
|
accounts. Adapted from patch submitted by Shad. [MN00039941]
|
||||||
|
|
||||||
|
* Wed Feb 23 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-30]
|
||||||
|
- Fix incompatibility with CentOS's CGI.pm.
|
||||||
|
|
||||||
|
* Tue Feb 22 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-29]
|
||||||
|
- Fix bug in output to empty file when FILTER is used
|
||||||
|
during template expansion. [charlieb MN00050075]
|
||||||
|
|
||||||
|
* Tue Feb 22 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-28]
|
||||||
|
- Refresh contents of /etc/sysconfig/hwconf before listing
|
||||||
|
network adaptors [MN00069993]
|
||||||
|
- Fix typo in documentation of esmith::DB::db - reported from Tanna -
|
||||||
|
http://www.livejournal.com/users/gcrumb/61169.html (thanks Dan!)
|
||||||
|
|
||||||
|
* Wed Feb 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-27]
|
||||||
|
- Fix typo. [MN00066059]
|
||||||
|
|
||||||
|
* Wed Feb 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-26]
|
||||||
|
- Use /sbin/e-smith/whiptail if it is available. [MN00066059]
|
||||||
|
|
||||||
|
* Mon Feb 7 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-25]
|
||||||
|
- Update perms and ownership when expanding templates, regardless of
|
||||||
|
whether file content has changed or not. [MN00068043]
|
||||||
|
|
||||||
|
* Wed Feb 2 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-24]
|
||||||
|
- Fix the insertion of implicit actions into sorted action list in
|
||||||
|
event.pm. Problem was perl syntax ambiguity. [MN00066406]
|
||||||
|
|
||||||
|
* Fri Jan 28 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-23]
|
||||||
|
- Really move /home/e-smith/* to e-smith-base. [MN00066635]
|
||||||
|
|
||||||
|
* Fri Jan 28 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-22]
|
||||||
|
- Move /home/e-smith/* to e-smith-base. [MN00066635]
|
||||||
|
- Move generic-template expand to S05 position in sort, and
|
||||||
|
adjust-service to S90. Fix run-time error. [MN00066406]
|
||||||
|
- Don't attempt to execute non-executable action scripts.
|
||||||
|
|
||||||
|
* Thu Jan 27 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-21]
|
||||||
|
- Implicitly include template expand and services adjust actions
|
||||||
|
in each event if the respective metadata directory exists.
|
||||||
|
[MN00066406]
|
||||||
|
|
||||||
|
* Tue Jan 25 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-20]
|
||||||
|
- Add "adjust" to list of verbs which serviceControl groks,
|
||||||
|
to allow for "masq adjust". [MN00065576]
|
||||||
|
|
||||||
|
* Tue Jan 25 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-19]
|
||||||
|
- Add adjust-services generic action script [MN00065576]
|
||||||
|
|
||||||
|
* Tue Jan 18 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-18]
|
||||||
|
- Fix typo. [MN00064412]
|
||||||
|
- Improve die() message in esmith::config::STORE. [MN00064394]
|
||||||
|
|
||||||
|
* Mon Jan 17 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-17]
|
||||||
|
- Fix broken logging (and reduce logging) in initialize_default_databases.
|
||||||
|
[MN00064412]
|
||||||
|
- Fix inappropriate use of global $_ in initialize_default_databases.
|
||||||
|
[MN00064415]
|
||||||
|
|
||||||
|
* Thu Dec 23 2004 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-16]
|
||||||
|
- Read templated file metadata (if any) from file tree under
|
||||||
|
/etc/e-smith/templates.metadata. Update expand-template to
|
||||||
|
use current API. [MN00061830]
|
||||||
|
|
||||||
|
* Tue Dec 14 2004 Mark Knox <markk@e-smith.com>
|
||||||
|
- [1.15.1-15]
|
||||||
|
- Change copyright date to 2004 [markk MN00060958]
|
||||||
|
|
||||||
|
* Fri Nov 5 2004 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-14]
|
||||||
|
- Fix Authen::PAM dependency header [charlieb MN00040240]
|
||||||
|
- Use kudzu's generated /etc/sysconfig/hwconf file for NIC detection
|
||||||
|
[charlieb MN00056220]
|
||||||
|
|
||||||
|
* Thu Oct 14 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-13]
|
||||||
|
- Updated esmith::ethernet's search code to remove File::Find, as it doesn't
|
||||||
|
get along with taint checking. [msoulier MN00052510]
|
||||||
|
|
||||||
|
* Wed Oct 13 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-12]
|
||||||
|
- Updated esmith::ethernet's search code such that is it more adaptable, and
|
||||||
|
recurses the directories given. [msoulier MN00052510]
|
||||||
|
|
||||||
|
* Wed Oct 13 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-11]
|
||||||
|
- Updated esmith::ethernet's search paths for network drivers.
|
||||||
|
[msoulier MN00052510]
|
||||||
|
|
||||||
|
* Mon Oct 4 2004 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-10]
|
||||||
|
- Remove dependency on perl(Filter::Handle) [charlieb MN00050075]
|
||||||
|
|
||||||
|
* Fri Sep 24 2004 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-09]
|
||||||
|
- Updated requires with new perl dependencies. [msoulier MN00040240]
|
||||||
|
- Remove "AutoReqProv: no" so that "Provides" headers are auto-generated.
|
||||||
|
[charlieb MN00040240]
|
||||||
|
- Remove anachronistic "require v5.6.0" directives. [charlieb MN00050370]
|
||||||
|
- Avoid use of Filter::Handle in esmith::template. [charlieb MN00050075]
|
||||||
|
|
||||||
|
* Fri Aug 27 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-08]
|
||||||
|
- Added esmith::util::network::isValidEmail function. [msoulier MN00023814]
|
||||||
|
|
||||||
|
* Thu Aug 26 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-07]
|
||||||
|
- Added svdisable to permissible actions in serviceControl.
|
||||||
|
[msoulier MN00043056]
|
||||||
|
|
||||||
|
* Tue Aug 10 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-06]
|
||||||
|
- Fixed new methods. Bad else case. [msoulier MN00044891]
|
||||||
|
|
||||||
|
* Fri Aug 6 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-05]
|
||||||
|
- Added a keys() method. [msoulier MN00041968]
|
||||||
|
|
||||||
|
* Fri Aug 6 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-04]
|
||||||
|
- Added set_prop and set_value methods in esmith::DB. [msoulier MN00044891]
|
||||||
|
|
||||||
|
* Tue Jul 20 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-03]
|
||||||
|
- Undeprecated esmith::util::serviceControl. [msoulier MN00043056]
|
||||||
|
|
||||||
|
* Fri Jun 25 2004 Tony Clayton <apc@e-smith.com>
|
||||||
|
- [1.15.1-02]
|
||||||
|
- Merge language_tag2locale() function from perl-I18N-LangTags [tonyc
|
||||||
|
MN00040170]
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-01]
|
||||||
|
- Rolling to collect patches.
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-23]
|
||||||
|
- Reordered the create code slightly to catch more errors.
|
||||||
|
[msoulier MN00035059]
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-22]
|
||||||
|
- Added yet more error handling and reporting. [msoulier MN00035059]
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-21]
|
||||||
|
- Fixed one $Error reference that I missed in the last rev.
|
||||||
|
[msoulier MN00035059]
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-20]
|
||||||
|
- Moved error handling from esmith::DB::db to esmith::DB, since it should not
|
||||||
|
be database implementation specific. [msoulier MN00035059]
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-19]
|
||||||
|
- Propagated migration failures up to calling code for reporting to syslog.
|
||||||
|
- Propagated creation failures up to calling code.
|
||||||
|
- Moved lexicon $Error in esmith::DB::db to a class property so it can be used
|
||||||
|
by subclasses. [msoulier MN00035059]
|
||||||
|
|
||||||
|
* Thu May 27 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-18]
|
||||||
|
- Changed print statements to calls to the logger. [msoulier MN00035059]
|
||||||
|
|
||||||
|
* Thu May 27 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-17]
|
||||||
|
- Added print statements to initialize-default-databases for post-install
|
||||||
|
debugging. [msoulier MN00035059]
|
||||||
|
|
||||||
|
* Fri May 7 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-16]
|
||||||
|
- Fixed migrate to report the caught error message. [msoulier MN00032503]
|
||||||
|
|
||||||
|
* Thu May 6 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-15]
|
||||||
|
- Added isValidHostname function to esmith::util::network.
|
||||||
|
[msoulier MN00024212]
|
||||||
|
|
||||||
|
* Fri Feb 13 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-14]
|
||||||
|
- Greatly simplified the _mysystem function by ripping out open3.
|
||||||
|
[msoulier dpar-20385]
|
||||||
|
|
||||||
|
* Fri Feb 13 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-13]
|
||||||
|
- Backed-out change in esmith::util, as it's non-trivial there with the exec.
|
||||||
|
Completed update of esmith::event, and separated the esmith::Logger class.
|
||||||
|
[msoulier dpar-20385]
|
||||||
|
|
||||||
|
* Fri Feb 13 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-12]
|
||||||
|
- Removed use of the logger entirely, replacing it with an abstracted
|
||||||
|
interface to the Sys::Syslog module. [msoulier dpar-20385]
|
||||||
|
|
||||||
|
* Thu Jan 8 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-11]
|
||||||
|
- Fixed esmith::util::network::isValidIP() so valid IP substrings no longer
|
||||||
|
return true values. [msoulier 9308]
|
||||||
|
|
||||||
|
* Thu Jan 8 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-10]
|
||||||
|
- Added a check in STORE in esmith::config for invalid attempts to set a value
|
||||||
|
without a type. Also escalated previous warnings for undef key and value to
|
||||||
|
fatal exceptions. [msoulier 7386]
|
||||||
|
|
||||||
|
* Thu Jan 8 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-09]
|
||||||
|
- Now trimming whitespace around keys and values during esmith::config STORE
|
||||||
|
events, to prevent invalid keys and values from being saved. [msoulier 7021]
|
||||||
|
|
||||||
|
* Mon Jan 5 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-08]
|
||||||
|
- Fixed POD around merge_props. [msoulier 9482]
|
||||||
|
|
||||||
|
* Fri Nov 7 2003 Tony Clayton <apc@e-smith.com>
|
||||||
|
- [1.15.0-07]
|
||||||
|
- And again [tonyc 10569]
|
||||||
|
|
||||||
|
* Fri Nov 7 2003 Tony Clayton <apc@e-smith.com>
|
||||||
|
- [1.15.0-06]
|
||||||
|
- Change Merge API a bit, fix pod [tonyc 10569]
|
||||||
|
|
||||||
|
* Fri Nov 7 2003 Tony Clayton <apc@e-smith.com>
|
||||||
|
- [1.15.0-05]
|
||||||
|
- Add esmith::DB::Merge library [tonyc 10569]
|
||||||
|
|
||||||
|
* Fri Oct 10 2003 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-04]
|
||||||
|
- Fixed AccountsDB.pm to handle group names with hyphens and periods, to match
|
||||||
|
the error message in the groups panel, and the rest of the group/user
|
||||||
|
behaviour. [msoulier 10236]
|
||||||
|
|
||||||
|
* Sun Sep 21 2003 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.0-03]
|
||||||
|
- Skip any directries when iterating over action scripts in esmith::event.
|
||||||
|
Add logfile timestamp symlinking to generic_template_expand action.
|
||||||
|
Use templates2expand and logfiles2timestamp subdirectories of the event
|
||||||
|
directory. Fix shebang line. [charlieb 10035]
|
||||||
|
|
||||||
|
* Thu Sep 18 2003 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.0-02]
|
||||||
|
- Add generic_template_expand action. [charlieb 10035]
|
||||||
|
|
||||||
|
* Thu Sep 18 2003 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.0-01]
|
||||||
|
- Changing version to development stream number - 1.15.0
|
||||||
|
|
||||||
|
%prep
|
||||||
|
%setup
|
||||||
|
|
||||||
|
%pre
|
||||||
|
#--------------------------------------------------
|
||||||
|
# add admin, public and www user accounts
|
||||||
|
#--------------------------------------------------
|
||||||
|
/usr/sbin/groupadd \
|
||||||
|
-g 500 shared 2>/dev/null || :
|
||||||
|
/usr/sbin/useradd \
|
||||||
|
-u 102 -c 'e-smith web server' -d /home/e-smith \
|
||||||
|
-G shared -M -s /bin/false www 2>/dev/null || :
|
||||||
|
/usr/sbin/useradd \
|
||||||
|
-u 101 -c 'e-smith administrator' -d /home/e-smith \
|
||||||
|
-G shared,root,www -M -s /sbin/e-smith/console admin 2>/dev/null || :
|
||||||
|
/usr/sbin/useradd \
|
||||||
|
-u 103 -c 'e-smith guest' -d /home/e-smith \
|
||||||
|
-G shared -M -s /bin/false public 2>/dev/null || :
|
||||||
|
/usr/sbin/useradd \
|
||||||
|
-u 1002 -c 'sme log user' -d /var/log/smelog \
|
||||||
|
-M -s /bin/false smelog 2>/dev/null || :
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
|
||||||
|
%post
|
||||||
|
|
||||||
|
%build
|
||||||
|
for event in post-install post-upgrade bootstrap-console-save console-save
|
||||||
|
do
|
||||||
|
mkdir -p root/etc/e-smith/events/$event
|
||||||
|
done
|
||||||
|
perl createlinks
|
||||||
|
|
||||||
|
%install
|
||||||
|
rm -rf $RPM_BUILD_ROOT
|
||||||
|
(cd root ; find . -depth -print | cpio -dump $RPM_BUILD_ROOT)
|
||||||
|
rm -f %{name}-%{version}-%{release}-filelist
|
||||||
|
/sbin/e-smith/genfilelist $RPM_BUILD_ROOT \
|
||||||
|
>%{name}-%{version}-%{release}-filelist
|
||||||
|
echo "%doc Copying" >> %{name}-%{version}-%{release}-filelist
|
||||||
|
echo "%doc Artistic" >> %{name}-%{version}-%{release}-filelist
|
||||||
|
echo "%doc LICENSE" >> %{name}-%{version}-%{release}-filelist
|
||||||
|
|
||||||
|
%clean
|
||||||
|
rm -rf $RPM_BUILD_ROOT
|
||||||
|
|
||||||
|
%files -f %{name}-%{version}-%{release}-filelist
|
||||||
|
%defattr(-,root,root)
|
1
contriborbase
Normal file
1
contriborbase
Normal file
@ -0,0 +1 @@
|
|||||||
|
sme10
|
38
createlinks
Normal file
38
createlinks
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright (c) 2001 Mitel Networks Corporation
|
||||||
|
#
|
||||||
|
# Technical support for this program is available from Mitel Networks
|
||||||
|
# Corporation. Please visit our web site www.e-smith.com for details.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
sub panel_link
|
||||||
|
{
|
||||||
|
my ($function, $panel) = @_;
|
||||||
|
|
||||||
|
unlink "root/etc/e-smith/web/panels/$panel/cgi-bin/$function";
|
||||||
|
symlink("../../../functions/$function",
|
||||||
|
"root/etc/e-smith/web/panels/$panel/cgi-bin/$function")
|
||||||
|
or die "Can't symlink to root/etc/e-smith/web/panels/$panel".
|
||||||
|
"/cgi-bin/$function: $!";
|
||||||
|
}
|
||||||
|
|
||||||
|
sub event_link
|
||||||
|
{
|
||||||
|
my ($action, $event, $level) = @_;
|
||||||
|
|
||||||
|
unlink "root/etc/e-smith/events/${event}/S${level}${action}";
|
||||||
|
symlink("../actions/${action}",
|
||||||
|
"root/etc/e-smith/events/${event}/S${level}${action}")
|
||||||
|
or die "Can't symlink to " .
|
||||||
|
"root/etc/e-smith/events/${event}/S${level}${action}: $!";
|
||||||
|
}
|
||||||
|
|
||||||
|
#--------------------------------------------------
|
||||||
|
# actions for events
|
||||||
|
#--------------------------------------------------
|
||||||
|
|
||||||
|
foreach my $event (qw(post-install post-upgrade bootstrap-console-save console-save))
|
||||||
|
{
|
||||||
|
event_link("initialize-default-databases", $event, "00");
|
||||||
|
}
|
902
e-smith-lib.spec
Normal file
902
e-smith-lib.spec
Normal file
@ -0,0 +1,902 @@
|
|||||||
|
# $Id: e-smith-lib.spec,v 1.20 2023/07/11 02:49:10 trevorb Exp $
|
||||||
|
|
||||||
|
%define copykooz 2013-2021
|
||||||
|
|
||||||
|
Summary: e-smith server and gateway - library module
|
||||||
|
%define name e-smith-lib
|
||||||
|
Name: %{name}
|
||||||
|
%define version 2.6.0
|
||||||
|
%define release 17
|
||||||
|
Version: %{version}
|
||||||
|
Release: %{release}%{?dist}
|
||||||
|
License: Artistic
|
||||||
|
Group: Networking/Daemons
|
||||||
|
Source: %{name}-%{version}.tar.xz
|
||||||
|
|
||||||
|
BuildRoot: /var/tmp/%{name}-%{version}-%{release}-buildroot
|
||||||
|
BuildArchitectures: noarch
|
||||||
|
BuildRequires: e-smith-devtools >= 1.6.3-01
|
||||||
|
Obsoletes: %{name}-Tai64n
|
||||||
|
Obsoletes: whiptail
|
||||||
|
Requires(pre): e-smith
|
||||||
|
Requires: dialog
|
||||||
|
Requires: perl, perl(Text::Template)
|
||||||
|
Requires: perl(Time::HiRes), perl(MIME::Base64)
|
||||||
|
Requires: perl(Authen::PAM), perl(I18N::AcceptLanguage)
|
||||||
|
Requires: perl(I18N::LangTags) >= 0.27
|
||||||
|
Requires: perl(Net::IPv4Addr) >= 0.10
|
||||||
|
Requires: perl(Taint::Util)
|
||||||
|
Requires: pciutils
|
||||||
|
|
||||||
|
%description
|
||||||
|
e-smith server and gateway software - library module.
|
||||||
|
|
||||||
|
%changelog
|
||||||
|
* Wed Jul 12 2023 cvs2git.sh aka Brian Read <brianr@koozali.org> 2.6.0-17.sme
|
||||||
|
- Roll up patches and move to git repo [SME: 12338]
|
||||||
|
- Remove create e-smith-lib-update event [SME: 12338]
|
||||||
|
|
||||||
|
* Wed Jul 12 2023 BogusDateBot
|
||||||
|
- Eliminated rpmbuild "bogus date" warnings due to inconsistent weekday,
|
||||||
|
by assuming the date is correct and changing the weekday.
|
||||||
|
|
||||||
|
* Wed Nov 23 2022 Jean-Philippe Pialasse <tests@pialasse.com> 2.6.0-16.sme
|
||||||
|
- allow call to systemctl aliases in etc [SME: 12194]
|
||||||
|
fix network=>networking service is not restarted.
|
||||||
|
|
||||||
|
* Thu Oct 28 2021 Jean-Philippe Pialasse <tests@pialasse.com> 2.6.0-15.sme
|
||||||
|
- add support for service with instances [SME: 11723]
|
||||||
|
|
||||||
|
* Mon Apr 19 2021 Jean-Philippe Pialasse <tests@pialasse.com> 2.6.0-14.sme
|
||||||
|
- update copyright dates, and make it easier to change from spec file [SME: 11570]
|
||||||
|
|
||||||
|
* Sat Jan 02 2021 Jean-Philipe Pialasse <tests@pialasse.com> 2.6.0-13.sme
|
||||||
|
- partial revert of signals [SME: 11177]
|
||||||
|
signal s not passed to runit services (dnscache*, qmail, qpsmtpd...)
|
||||||
|
services handled by systemd crash if they do not have Restart=always defined
|
||||||
|
|
||||||
|
* Fri Jan 01 2021 Jean-Philipe Pialasse <tests@pialasse.com> 2.6.0-12.sme
|
||||||
|
- add support for signals SIG* with systemd [SME: 11177]
|
||||||
|
fix typo for reload-or-try-restart
|
||||||
|
unsupervised services: really stop when disabled and start stopped enabled ones
|
||||||
|
|
||||||
|
* Thu Nov 26 2020 Jean-Philipe Pialasse <tests@pialasse.com> 2.6.0-11.sme
|
||||||
|
- remove error when sending sighup event [SME: 11177]
|
||||||
|
|
||||||
|
* Tue Nov 17 2020 Jean-Philipe Pialasse <tests@pialasse.com> 2.6.0-10.sme
|
||||||
|
- allow more systemctl controls [SME: 11177]
|
||||||
|
convert unrecognized signals from service2adjust in events for systemd
|
||||||
|
handle unsupervised services the same way supervised were in adjust-services
|
||||||
|
- create e-smith-lib-event [SME: 11141]
|
||||||
|
|
||||||
|
* Thu Dec 12 2019 Jean-Philipe Pialasse <tests@pialasse.com> 2.6.0-9.sme
|
||||||
|
- add support for systemctl reload-or-restart, try-restart, enable -now [SME: 10848]
|
||||||
|
|
||||||
|
* Wed Oct 25 2017 Daniel Berteaud <daniel@firewall-services.com> 2.6.0-8.sme
|
||||||
|
- Revert prev change regarding tap_soft, as it's not needed [SME: 10445]
|
||||||
|
|
||||||
|
* Mon Oct 23 2017 Daniel Berteaud <daniel@firewall-services.com> 2.6.0-7.sme
|
||||||
|
- Skip tap_soft interfaces (eg SoftEther, code from Hsing-Foo Wang)
|
||||||
|
[SME: 10445]
|
||||||
|
|
||||||
|
* Wed Apr 12 2017 Jean-Philipe Pialasse <tests@pialasse.com> 2.6.0-6.sme
|
||||||
|
- remove reference to smtpd in configuration.conf [SME: 9478]
|
||||||
|
|
||||||
|
* Thu Aug 04 2016 Jean-Philipe Pialasse <tests@pialasse.com> 2.6.0-5.sme
|
||||||
|
- fix console startup display [SME: 9352]
|
||||||
|
|
||||||
|
* Sat Jul 23 2016 Jean-Philipe Pialasse <tests@pialasse.com> 2.6.0-4.sme
|
||||||
|
- fix service name syslog to rsyslog [SME: 9691]
|
||||||
|
|
||||||
|
* Mon Jul 18 2016 Jean-Philipe Pialasse <tests@pialasse.com> 2.6.0-3.sme
|
||||||
|
- fix mysqld to mariadb [SME: 9438]
|
||||||
|
- Eliminated rpmbuild "bogus date" warnings due to inconsistent weekday,
|
||||||
|
by assuming the date is correct and changing the weekday.
|
||||||
|
Sat Dec 25 2005 --> Sat Dec 24 2005 or Sun Dec 25 2005 or Sat Dec 31 2005 or ....
|
||||||
|
Tue Dec 15 2006 --> Tue Dec 12 2006 or Fri Dec 15 2006 or Tue Dec 19 2006 or ....
|
||||||
|
|
||||||
|
* Thu Jul 14 2016 stephane de Labrusse <stephdl@de-labrusse.fr> 2.6.0-2.sme
|
||||||
|
- fix esmith::util::serviceControl to manage systemd service [SME: 9660]
|
||||||
|
- Added e-smith-lib-2.6.0.bz9660.serviceControlSystemd.patch
|
||||||
|
|
||||||
|
* Fri Feb 05 2016 stephane de Labrusse <stephdl@de-labrusse.fr> 2.6.0-1.sme
|
||||||
|
- Initial release to sme10
|
||||||
|
|
||||||
|
* Tue Jan 12 2016 Daniel Berteaud <daniel@firewall-services.com> 2.4.0-16.sme
|
||||||
|
- Adjust regex so adjust-service accepts sigusr1 and sigusr2 from files
|
||||||
|
[SME: 9184]
|
||||||
|
|
||||||
|
* Thu Jun 18 2015 stephane de Labrusse <stephdl@de-labrusse.fr> 2.4.0-15.sme
|
||||||
|
- Added a password creation (set_secret) [SME: 8943]
|
||||||
|
- code from Charlie Brady <charlieb-contribs-bugzilla@budge.apana.org.au>
|
||||||
|
|
||||||
|
* Sat Apr 5 2014 Ian Wells <esmith@wellsi.com> 2.4.0-14.sme
|
||||||
|
- Untaint the license filename [SME: 8305]
|
||||||
|
- Non-functional perl::Critic changes.
|
||||||
|
|
||||||
|
* Sun Mar 16 2014 Ian Wells <esmith@wellsi.com> 2.4.0-13.sme
|
||||||
|
- Add textbox() to console.pm, getLicenseFile to util.pm [SME: 8264]
|
||||||
|
|
||||||
|
* Sun Feb 2 2014 Ian Wells <esmith@wellsi.com> 2.4.0-12.sme
|
||||||
|
- Update frame header and footer [SME: 8183]
|
||||||
|
|
||||||
|
* Sat Nov 30 2013 Daniel Berteaud <daniel@firewall-services.com> 2.4.0-11.sme
|
||||||
|
- Remove the "swap interface" feature [SME: 7993]
|
||||||
|
|
||||||
|
* Sat Nov 30 2013 Daniel Berteaud <daniel@firewall-services.com> 2.4.0-10.sme
|
||||||
|
- Return nic names in probeAdapters so we can drop HWAddress [SME: 7991]
|
||||||
|
|
||||||
|
* Sat May 25 2013 Ian Wells <esmith@wellsi.com> 2.4.0-9.sme
|
||||||
|
- Correctly display accented letters in the console [SME: 7591]
|
||||||
|
by Filippo Carletti
|
||||||
|
|
||||||
|
* Sun May 05 2013 Ian Wells <esmith@wellsi.com> 2.4.0-8.sme
|
||||||
|
- Add e-smith as a Requires(pre) and remove adding users in %pre
|
||||||
|
- Fix uid and gid to be the same in create-system-user [SME: 7547]
|
||||||
|
|
||||||
|
* Sat Mar 9 2013 Shad L. Lords <slords@mail.com> 2.4.0-7.sme
|
||||||
|
- Ensure group www exists before user [SME: 7319]
|
||||||
|
|
||||||
|
* Wed Mar 6 2013 Shad L. Lords <slords@mail.com> 2.4.0-6.sme
|
||||||
|
- Fix pre script to make www and apache the same [SME: 7319]
|
||||||
|
|
||||||
|
* Tue Mar 5 2013 Daniel Berteaud <daniel@firewall-services.com> 2.4.0-5.sme
|
||||||
|
- Untaint variable in probeAdapters() [SME: 7416]
|
||||||
|
|
||||||
|
* Thu Feb 28 2013 Ian Wells <esmith@wellsi.com> 2.4.0-4.sme
|
||||||
|
- Improve error checking in isValidIP [SME: 7410]
|
||||||
|
|
||||||
|
* Sun Feb 24 2013 Daniel Berteaud <daniel@firewall-services.com> 2.4.0-3.sme
|
||||||
|
- Fix MAC detection for bond slaves [SME: 3596]
|
||||||
|
|
||||||
|
* Thu Jan 31 2013 Daniel Berteaud <daniel@firewall-services.com> 2.4.0-2.sme
|
||||||
|
- Stop using kudzu for NIC detection [SME: 3596]
|
||||||
|
|
||||||
|
* Fri Jan 25 2013 Shad L. Lords <slords@mail.com> 2.4.0-1.sme
|
||||||
|
- Bump version in prep for SME9
|
||||||
|
- Perl module rpms moved to /usr/share/perl5/vendor_perl [SME: 7223]
|
||||||
|
|
||||||
|
* Mon Oct 11 2010 Shad L. Lords <slords@mail.com> 2.2.0-7.sme
|
||||||
|
- Serialize configure_peers to prevent errors [SME: 5831]
|
||||||
|
|
||||||
|
* Fri Feb 5 2010 Stephen Noble <support@dungog.net> 2.2.0-6.sme
|
||||||
|
- adds the hwaddr parameter to probeAdapters() [SME: 4528]
|
||||||
|
|
||||||
|
* Thu Feb 4 2010 Shad L. Lords <slords@mail.com> 2.2.0-5.sme
|
||||||
|
- Update path for 64-bit compatibility [SME: 5756]
|
||||||
|
|
||||||
|
* Tue Dec 22 2009 Filippo Carletti <filippo.carletti@gmail.com> 2.2.0-4.sme
|
||||||
|
- Really apply previous patch in the spec file. [SME: 5659]
|
||||||
|
|
||||||
|
* Wed Dec 9 2009 Charlie Brady <charlieb@budge.apana.org.au> 2.2.0-3.sme
|
||||||
|
- Add patch (Federico Simoncelli) to prevent re-use of uids. [SME: 5659]
|
||||||
|
|
||||||
|
* Mon Oct 13 2008 Shad L. Lords <slords@mail.com> 2.2.0-2.sme
|
||||||
|
- Add patch to support multiple samba roles [SME: 4172]
|
||||||
|
|
||||||
|
* Tue Oct 7 2008 Shad L. Lords <slords@mail.com> 2.2.0-1.sme
|
||||||
|
- Roll new stream to separate sme7/sme8 trees [SME: 4633]
|
||||||
|
|
||||||
|
* Wed Aug 20 2008 Shad L. Lords <slords@mail.com> 1.19.0-1
|
||||||
|
- Roll new dev stream.
|
||||||
|
|
||||||
|
* Sat Aug 9 2008 Shad L. Lords <slords@mail.com> 1.18.0-28
|
||||||
|
- Read /dev/urandmon instead of /dev/random [SME: 4492]
|
||||||
|
|
||||||
|
* Sat Aug 9 2008 Shad L. Lords <slords@mail.com> 1.18.0-27
|
||||||
|
- Make dialog not use STDERR for return values [SME: 3663]
|
||||||
|
- Remove requires for whiptail. No longer needed [SME: 4491]
|
||||||
|
|
||||||
|
* Thu Jul 31 2008 Shad L. Lords <slords@mail.com> 1.18.0-26
|
||||||
|
- Make binmode properties of db class [SME: 4317]
|
||||||
|
- Add new navigation db & utf8 classes [SME: 4317]
|
||||||
|
|
||||||
|
* Wed Apr 2 2008 Shad L. Lords <slords@mail.com> 1.18.0-25
|
||||||
|
- Add .utf8 to locale unless server-console [SME: 4162]
|
||||||
|
- Add new languages to langtag2locale [SME: 4163]
|
||||||
|
|
||||||
|
* Sun Mar 30 2008 Shad L. Lords <slords@mail.com> 1.18.0-24
|
||||||
|
- Gracefully handle encoding errors in navigation dbs [SME: 4147]
|
||||||
|
|
||||||
|
* Wed Mar 26 2008 Shad L. Lords <slords@mail.com> 1.18.0-23
|
||||||
|
- Make PAM conv routine not look for english strings [SME: 4117]
|
||||||
|
|
||||||
|
* Tue Mar 25 2008 Shad L. Lords <slords@mail.com> 1.18.0-22
|
||||||
|
- Fix wide output to syslog and allow navigations db to be utf8 [SME: 4101]
|
||||||
|
|
||||||
|
* Mon Mar 10 2008 Shad L. Lords <slords@mail.com> 1.18.0-21
|
||||||
|
- Output navigation in UTF-8 [SME: 3858]
|
||||||
|
|
||||||
|
* Sun Jan 20 2008 Gavin Weight <gweight@gmail.com> 1.18.0-20
|
||||||
|
- Call smbpasswd -e - to reset user correctly. [SME: 3755]
|
||||||
|
|
||||||
|
* Tue Jan 08 2008 Stephen Noble <support@dungog.net> 1.18.0-19
|
||||||
|
- modify validate password strong match for console [SME: 2173]
|
||||||
|
|
||||||
|
* Thu Nov 01 2007 Gavin Weight <gweight@gmail.com> 1.18.0-18
|
||||||
|
- Fix I18N.pm file descriptor leak. [SME: 3509]
|
||||||
|
|
||||||
|
* Wed Oct 31 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-17
|
||||||
|
- Remove undocumented and unused template metadata handling from
|
||||||
|
generic_template_expand action (Take II). [SME: 2798]
|
||||||
|
|
||||||
|
* Fri Jun 1 2007 Shad L. Lords <slords@mail.com> 1.18.0-16
|
||||||
|
- pam_unix requires passwords >= 6 [SME: 3039]
|
||||||
|
|
||||||
|
* Mon May 21 2007 Shad L. Lords <slords@mail.com> 1.18.0-15
|
||||||
|
- Fix ip addr shift to work on 64-bit
|
||||||
|
|
||||||
|
* Sun Apr 29 2007 Shad L. Lords <slords@mail.com>
|
||||||
|
- Clean up spec so package can be built by koji/plague
|
||||||
|
|
||||||
|
* Thu Apr 05 2007 Shad L. Lords <slords@mail.com> 1.18.0-14
|
||||||
|
- Reverse changes for 1.18.0-12 as it is being used. [SME: 2838]
|
||||||
|
|
||||||
|
* Wed Apr 04 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-13
|
||||||
|
- Allow gauge console widget to be used without following message_page.
|
||||||
|
[SME: 2832].
|
||||||
|
|
||||||
|
* Mon Mar 26 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-12
|
||||||
|
- Remove undocumented and unused template metadata handling from
|
||||||
|
generic_template_expand action. [SME: 2798]
|
||||||
|
|
||||||
|
* Mon Mar 26 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-11
|
||||||
|
- Split logfile rotation code out of generic_template_expand.
|
||||||
|
Code moved to a new action in e-smith-base. [SME: 2795]
|
||||||
|
|
||||||
|
* Mon Mar 19 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-10
|
||||||
|
Don't try to read template.metadata from subdirectories (affects
|
||||||
|
initialize_database_defaults). [SME: 2690]
|
||||||
|
|
||||||
|
* Fri Mar 16 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-9
|
||||||
|
- Allow template.metadata to be provided via a directory of
|
||||||
|
small files. [SME: 2690]
|
||||||
|
- Add support for a DELETE boolean template.metadata item. [SME: 2691]
|
||||||
|
|
||||||
|
* Wed Feb 28 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-8
|
||||||
|
- Add gauge widget to esmith::console. [SME: 2579]
|
||||||
|
|
||||||
|
* Sun Feb 25 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-7
|
||||||
|
- Default infobox height to 8 rows, and allow override via params.
|
||||||
|
[SME: 2560]
|
||||||
|
|
||||||
|
* Fri Feb 23 2007 Shad L. Lords <slords@mail.com> 1.18.0-6
|
||||||
|
- Really fix clear parameter for inputbox dialog screen [SME: 2533]
|
||||||
|
|
||||||
|
* Fri Feb 23 2007 Shad L. Lords <slords@mail.com> 1.18.0-5
|
||||||
|
- Fix clear parameter for dialog screens [SME: 2533]
|
||||||
|
|
||||||
|
* Thu Feb 22 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-4
|
||||||
|
- switch esmith::console::password_page() to use dialog rather than
|
||||||
|
whiptail. [SME: 2534]
|
||||||
|
|
||||||
|
* Thu Feb 22 2007 Charlie Brady <charlie_brady@mitel.com> 1.18.0-3
|
||||||
|
- Add infobox widget. [SME: 2533]
|
||||||
|
|
||||||
|
* Sun Jan 28 2007 Shad L. Lords <slords@mail.com> 1.18.0-2
|
||||||
|
- Fix backtitle spacing for new dialog menus [SME: 2328]
|
||||||
|
|
||||||
|
* Fri Jan 26 2007 Shad L. Lords <slords@mail.com> 1.18.0-1
|
||||||
|
- Roll stable stream. [SME: 2328]
|
||||||
|
|
||||||
|
* Fri Dec 15 2006 Federico Simoncelli <federico.simoncelli@gmail.com> 1.17.0-8
|
||||||
|
Tue Dec 15 2006 --> Tue Dec 12 2006 or Fri Dec 15 2006 or Tue Dec 19 2006 or ....
|
||||||
|
- Added the validatePassword function to esmith::util. [SME: 2100]
|
||||||
|
|
||||||
|
* Thu Dec 07 2006 Shad L. Lords <slords@mail.com>
|
||||||
|
- Update to new release naming. No functional changes.
|
||||||
|
- Make Packager generic
|
||||||
|
|
||||||
|
* Fri Nov 24 2006 Gordon Rowell <gordonr@gormand.com.au> 1.17.0-06
|
||||||
|
- Allow optional path to peeers directory in
|
||||||
|
esmith::tcpsvd::configure_peers() [SME: 2086]
|
||||||
|
|
||||||
|
* Tue Nov 14 2006 Charlie Brady <charlie_brady@mitel.com> 1.17.0-05
|
||||||
|
- Allow defaulno option to esmith::console::yesno.
|
||||||
|
- Fix I/O redirection in esmith::console::new.
|
||||||
|
|
||||||
|
* Tue Nov 07 2006 Charlie Brady <charlie_brady@mitel.com> 1.17.0-04
|
||||||
|
- Allow display of "0" in iesmith::cgi::genSmallCell. [SME: 2038]
|
||||||
|
|
||||||
|
* Mon Oct 23 2006 Charlie Brady <charlie_brady@mitel.com> 1.17.0-03
|
||||||
|
- Make dialog the default console app, with whiptail used only when
|
||||||
|
required.
|
||||||
|
|
||||||
|
* Wed Oct 11 2006 Charlie Brady <charlie_brady@mitel.com> 1.17.0-02
|
||||||
|
- Allow 'dialog' to be called from esmith::console. [SME: 1958]
|
||||||
|
|
||||||
|
* Wed Oct 11 2006 Charlie Brady <charlie_brady@mitel.com> 1.17.0-01
|
||||||
|
- Roll development branch.
|
||||||
|
|
||||||
|
* Mon Sep 25 2006 Charlie Brady <charlie_brady@mitel.com> 1.16.0-06
|
||||||
|
- Fix problem with greedy RE in template.metadata parsing. [SME: 1906]
|
||||||
|
|
||||||
|
* Fri Sep 08 2006 Charlie Brady <charlie_brady@mitel.com> 1.16.0-05
|
||||||
|
- Fix taint problem in template.metadata handling. [SME: 1906]
|
||||||
|
|
||||||
|
* Thu Apr 13 2006 Charlie Brady <charlie_brady@mitel.com> 1.16.0-04
|
||||||
|
- Use "kudzu --probe --class network" for NIC detection. [SME: 727]
|
||||||
|
|
||||||
|
* Fri Apr 7 2006 Gordon Rowell <gordonr@gormand.com.au> 1.16.0-03
|
||||||
|
- Don't call smbpasswd -e - setting the password is sufficient [SME: 1193]
|
||||||
|
|
||||||
|
* Tue Mar 28 2006 Gordon Rowell <gordonr@gormand.com.au> 1.16.0-02
|
||||||
|
- Log previous contents of db entry in DELETE log [SME: 1066]
|
||||||
|
|
||||||
|
* Tue Mar 14 2006 Charlie Brady <charlie_brady@mitel.com> 1.16.0-01
|
||||||
|
- Roll to stable stream version number. [SME: 1016]
|
||||||
|
|
||||||
|
* Fri Mar 10 2006 Charlie Brady <charlie_brady@mitel.com> 1.15.4-02
|
||||||
|
- Suppress warning from genSmallCell if text is undefined. [SME: 986]
|
||||||
|
|
||||||
|
* Fri Feb 17 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.4-01
|
||||||
|
- Roll patches up to 1.15.3-42
|
||||||
|
- Trim changelog prior to 1.15.0-01 [SME: 828]
|
||||||
|
|
||||||
|
* Thu Feb 16 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-42
|
||||||
|
- Adjust console title bar to 'SME Server' [SME: 726]
|
||||||
|
|
||||||
|
* Tue Feb 14 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-41
|
||||||
|
- Reworded text for template-begin and change URL to /development/
|
||||||
|
rather than /custom/ [SME: 773]
|
||||||
|
|
||||||
|
* Mon Feb 13 2006 Charlie Brady <charlie_brady@mitel.com> 1.15.3-40
|
||||||
|
- Update URL in default template-begin fragment. [SME: 773]
|
||||||
|
|
||||||
|
* Sat Feb 11 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-39
|
||||||
|
- Remove obsolete e-smith-lib-Tai64n package. [SME: 689]
|
||||||
|
|
||||||
|
* Sat Feb 11 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-38
|
||||||
|
- [Null changelog for missing version - we accidentally skipped
|
||||||
|
this version.]
|
||||||
|
|
||||||
|
* Sat Feb 11 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-37
|
||||||
|
- Fix get_all_by_prop in scalar context. [SME: 669,721]
|
||||||
|
|
||||||
|
* Mon Feb 6 2006 Shad L. Lords <slords@mail.com> 1.15.3-37
|
||||||
|
- Add ability to pass many props to get_all_by_prop [SME: 669]
|
||||||
|
|
||||||
|
* Mon Jan 23 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-36
|
||||||
|
- Fix warning during pseudonym deletion. [SME: 491]
|
||||||
|
|
||||||
|
* Fri Jan 20 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-35
|
||||||
|
- Fix up use of Sys::Syslog::syslog. [SME: 526]
|
||||||
|
|
||||||
|
* Thu Jan 19 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-34
|
||||||
|
- Remove running of kudzu during NIC probing. TODO: Find a solution
|
||||||
|
to system reconfiguration when new hardware is added. [SME: 192]
|
||||||
|
|
||||||
|
* Tue Jan 10 2006 Charlie Brady <charlieb@e-smith.com> 1.15.3-33
|
||||||
|
- Fold a.b.c.d/255.255.255.255 to a.b.c.d in local_access_spec() to
|
||||||
|
work around bugs in applications which don't accept such specs.
|
||||||
|
[SME: 430]
|
||||||
|
|
||||||
|
* Mon Jan 9 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-32
|
||||||
|
- Check whether an accounts db record exists before trying to create
|
||||||
|
the dot and underscore pseudonyms (new_record will fail silently)
|
||||||
|
and check that the records are pseudonyms before deleting them [SME: 24]
|
||||||
|
|
||||||
|
* Mon Jan 9 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-31
|
||||||
|
- And update POD for last change [SME: 24]
|
||||||
|
|
||||||
|
* Mon Jan 9 2006 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-30
|
||||||
|
- Allow dot and underscore in account names [SME: 24]
|
||||||
|
|
||||||
|
* Tue Dec 27 2005 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-29
|
||||||
|
- Remove newlines from cluck() string and just note that the old
|
||||||
|
path was used [SME: 365]
|
||||||
|
|
||||||
|
* Sun Dec 25 2005 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-28
|
||||||
|
Sat Dec 25 2005 --> Sat Dec 24 2005 or Sun Dec 25 2005 or Sat Dec 31 2005 or ....
|
||||||
|
- If db exists in both the old and new locations in
|
||||||
|
initialize_default_databases, rename the one in the new
|
||||||
|
location to db.time(), avoiding the conflict and saving the
|
||||||
|
evidence in case it is needed later [SME: 229]
|
||||||
|
|
||||||
|
* Mon Dec 05 2005 Mark Knox <mark_knox@mitel.com>
|
||||||
|
- [1.15.3-27]
|
||||||
|
- Moved queueing logic to separate daemon, replaced with FIFO IPC [BZ252
|
||||||
|
|
||||||
|
* Thu Dec 01 2005 Mark Knox <mark_knox@mitel.com>
|
||||||
|
- [1.15.3-26]
|
||||||
|
- Added event queueing (open source portion) for clustered systems [BZ250]
|
||||||
|
|
||||||
|
* Wed Nov 30 2005 Gordon Rowell <gordonr@gormand.com.au> 1.15.3-25
|
||||||
|
- Bump release number only
|
||||||
|
|
||||||
|
* Thu Nov 24 2005 Gordon Rowell <gordonr@e-smith.com>
|
||||||
|
- [1.15.3-24]
|
||||||
|
- Add missing 'use Locale::gettext' to esmith::console.pm [MN00108804]
|
||||||
|
|
||||||
|
* Sun Nov 20 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-23]
|
||||||
|
- Clarify logic for stopped services in adjust-services. [SF: 1357629]
|
||||||
|
|
||||||
|
* Sun Nov 20 2005 Gordon Rowell <gordonr@e-smith.com>
|
||||||
|
- [1.15.3-22]
|
||||||
|
- Correct adjust-services logic for stopped services [SF: 1357629]
|
||||||
|
|
||||||
|
* Wed Nov 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-21]
|
||||||
|
- Allow services2adjust directories to contain files rather than just dangling
|
||||||
|
symlinks. Files can contain more than one actions to perform. [SF: 1270644]
|
||||||
|
|
||||||
|
* Wed Nov 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-20]
|
||||||
|
- Also don't start services if we just want to "once" them. [SF: 1357629]
|
||||||
|
|
||||||
|
* Wed Nov 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-19]
|
||||||
|
- Fix restart of enabled supervised services which we are attempting to stop.
|
||||||
|
[SF: 1357629]
|
||||||
|
|
||||||
|
* Tue Nov 15 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-18]
|
||||||
|
- Set default for $type in esmith::cgi::genSmallCell, to prevent some log
|
||||||
|
noise. [SF: 1357830]
|
||||||
|
|
||||||
|
* Tue Nov 15 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-17]
|
||||||
|
- Pass $EVENT to template expansions in generic_template_expand.
|
||||||
|
[SF: MN00106104]
|
||||||
|
|
||||||
|
* Tue Nov 15 2005 Gordon Rowell <gordonr@e-smith.com>
|
||||||
|
- [1.15.3-16]
|
||||||
|
- Redirect esmith::config calls on old db paths to the new
|
||||||
|
location [SF: 1335865]
|
||||||
|
|
||||||
|
* Thu Oct 27 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-15]
|
||||||
|
- Fix a few minor spec file portability issues. [SF: 1339729]
|
||||||
|
|
||||||
|
* Wed Oct 26 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-14]
|
||||||
|
- Add it and de to the langtag2locale fixups. [SF: 1338236]
|
||||||
|
|
||||||
|
* Tue Oct 11 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-13]
|
||||||
|
- Build driver list from .ko files as well as .o files. Untaint driver
|
||||||
|
names while building list. [SF: 1323270]
|
||||||
|
|
||||||
|
* Mon Sep 26 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-12]
|
||||||
|
- Fix "defaults" handling so that values which evaluate to false are
|
||||||
|
not overridden by default. [SF: 1303885]
|
||||||
|
|
||||||
|
* Fri Sep 23 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-11]
|
||||||
|
- Untaint db names before attempting to move them. [MN00098405]
|
||||||
|
|
||||||
|
* Thu Sep 22 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-10]
|
||||||
|
- Provide networks method in esmith::NetworksDB. [SF: 1296099]
|
||||||
|
- Support a "localhost" configuration in esmith::tcpsvd:configure_peers
|
||||||
|
[SF: 1294719]
|
||||||
|
|
||||||
|
* Tue Sep 20 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-09]
|
||||||
|
- Remove deprecated functions from esmith::utils. [SF: 1295851]
|
||||||
|
- Include only "network" records in local_access_spec. [SF: 1296099]
|
||||||
|
|
||||||
|
* Mon Sep 12 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-08]
|
||||||
|
- Remove warning about explicit path in esmith::db::_db_path.
|
||||||
|
[SF: 1286294]
|
||||||
|
|
||||||
|
* Fri Sep 9 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-07]
|
||||||
|
- Tolerate, but warn about, symlinks in /home/e-smith. [SF: 1216546]
|
||||||
|
|
||||||
|
* Fri Sep 9 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-06]
|
||||||
|
- Reduce the noise from _file_path() in esmith::DB::db. [SF: 1286294]
|
||||||
|
|
||||||
|
* Wed Sep 7 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-05]
|
||||||
|
- Fix operation of expandTemplate when taint check is enabled.
|
||||||
|
[SF: 1284301]
|
||||||
|
|
||||||
|
* Wed Aug 17 2005 Mark Knox <markk@e-smith.com>
|
||||||
|
- [1.15.3-04]
|
||||||
|
- Added open_local and open_ro_local methods for clustering [markk MN00094831]
|
||||||
|
|
||||||
|
* Tue Aug 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-03]
|
||||||
|
- Fix POD error in util.pm.
|
||||||
|
|
||||||
|
* Wed Jul 27 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-02]
|
||||||
|
- Move databases default location from /home/e-smith to /home/e-smith/db.
|
||||||
|
During esmith::utils::initialize_default_databases, move from old to new
|
||||||
|
location before doing db migrate actions. [SF: 1216546]
|
||||||
|
|
||||||
|
* Wed Jul 27 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.3-01]
|
||||||
|
- Roll a new development stream - 1.15.3
|
||||||
|
|
||||||
|
* Wed Jul 27 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.2-04]
|
||||||
|
- Remove broken MergeDB stuff. [SF: 1246315]
|
||||||
|
|
||||||
|
* Wed Jul 27 2005 Mark Knox <markk@e-smith.com>
|
||||||
|
- [1.15.2-03]
|
||||||
|
- Fixed misleading comment in util.pm
|
||||||
|
- Added 'use' statements in Record classes for SOAP compatibility
|
||||||
|
- Fixed undefined max_len warning
|
||||||
|
- Added explicit writeconf calls in db::Record.pm, needed for setting props
|
||||||
|
via SOAP
|
||||||
|
|
||||||
|
* Tue Jul 19 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.2-02]
|
||||||
|
- Allow db open API to use default path if a simple filename
|
||||||
|
is provided as arg. This is preparation for move of dbs to
|
||||||
|
/home/e-smith/db. Patch by Shad Lords.
|
||||||
|
|
||||||
|
* Mon Jul 18 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.2-01]
|
||||||
|
- Roll new development stream - 1.15.2
|
||||||
|
|
||||||
|
* Fri Jul 15 2005 Mark Knox <markk@e-smith.com>
|
||||||
|
- [1.15.1-44]
|
||||||
|
- Tweak to allow calling _writeconf from SOAP [markk MN00090738]
|
||||||
|
|
||||||
|
* Tue Jun 21 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-43]
|
||||||
|
- Ensure that esmith::util::LdapPassword returns bare string without
|
||||||
|
newline terminator.
|
||||||
|
|
||||||
|
* Sun Jun 12 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-42]
|
||||||
|
- Remove .rpmsave and .rpmnew symlinks (as well as files). [SF: 1217969]
|
||||||
|
- Handle missing description in pcitables entries.
|
||||||
|
|
||||||
|
* Sun Jun 12 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-41]
|
||||||
|
- Provide feedback (via log messages) from services2adjust. [SF: 1218920]
|
||||||
|
|
||||||
|
* Mon May 30 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-40]
|
||||||
|
- Add preinstall scripting to create required accounts/groups if they
|
||||||
|
don't already exist. [SF: 1210723]
|
||||||
|
|
||||||
|
* Thu May 5 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-39]
|
||||||
|
- Show Text::Template error text rather than inappropriate $! if template
|
||||||
|
expansion fails.
|
||||||
|
- Change error to warning if a config item is set with an empty "type"
|
||||||
|
property.
|
||||||
|
|
||||||
|
* Thu May 5 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-38]
|
||||||
|
- Fix esmith::DB::get_prop_and_delete fail if prop is "false" [From Gordon].
|
||||||
|
|
||||||
|
* Tue May 3 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-37]
|
||||||
|
- Update directory list so that ethernet drivers from kernel-unsupported are
|
||||||
|
added to "choose by driver" list.
|
||||||
|
|
||||||
|
* Sat Mar 19 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-36]
|
||||||
|
- Rework esmith::tcpsvd::configure_peers so that it doesn't chdir.
|
||||||
|
- Fix generic_template_expand so that templates.metadata doesn't
|
||||||
|
need to set OUTPUT_FILENAME is TEMPLATE_PATH is changed. This
|
||||||
|
matches what expand-template already does.
|
||||||
|
|
||||||
|
* Fri Mar 18 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-35]
|
||||||
|
- Change numerous calls to "croak" in esmith::template
|
||||||
|
to "carp ...; return", so that problem with any single
|
||||||
|
template expansion doesn't terminate calling program.
|
||||||
|
[MN00075009]
|
||||||
|
|
||||||
|
* Wed Mar 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-34]
|
||||||
|
- Add missing "use esmith::util" in esmith::tcpsvd.
|
||||||
|
|
||||||
|
* Wed Mar 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-33]
|
||||||
|
- Add esmith::tcpsvd library for managing tcpsvd "peers"
|
||||||
|
directories.
|
||||||
|
|
||||||
|
* Mon Mar 14 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-32]
|
||||||
|
- Make template expansion message more succinct.
|
||||||
|
|
||||||
|
* Thu Mar 10 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-31]
|
||||||
|
- Remove pseudonyms of pseudonyms when removing user
|
||||||
|
accounts. Adapted from patch submitted by Shad. [MN00039941]
|
||||||
|
|
||||||
|
* Wed Feb 23 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-30]
|
||||||
|
- Fix incompatibility with CentOS's CGI.pm.
|
||||||
|
|
||||||
|
* Tue Feb 22 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-29]
|
||||||
|
- Fix bug in output to empty file when FILTER is used
|
||||||
|
during template expansion. [charlieb MN00050075]
|
||||||
|
|
||||||
|
* Tue Feb 22 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-28]
|
||||||
|
- Refresh contents of /etc/sysconfig/hwconf before listing
|
||||||
|
network adaptors [MN00069993]
|
||||||
|
- Fix typo in documentation of esmith::DB::db - reported from Tanna -
|
||||||
|
http://www.livejournal.com/users/gcrumb/61169.html (thanks Dan!)
|
||||||
|
|
||||||
|
* Wed Feb 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-27]
|
||||||
|
- Fix typo. [MN00066059]
|
||||||
|
|
||||||
|
* Wed Feb 16 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-26]
|
||||||
|
- Use /sbin/e-smith/whiptail if it is available. [MN00066059]
|
||||||
|
|
||||||
|
* Mon Feb 7 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-25]
|
||||||
|
- Update perms and ownership when expanding templates, regardless of
|
||||||
|
whether file content has changed or not. [MN00068043]
|
||||||
|
|
||||||
|
* Wed Feb 2 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-24]
|
||||||
|
- Fix the insertion of implicit actions into sorted action list in
|
||||||
|
event.pm. Problem was perl syntax ambiguity. [MN00066406]
|
||||||
|
|
||||||
|
* Fri Jan 28 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-23]
|
||||||
|
- Really move /home/e-smith/* to e-smith-base. [MN00066635]
|
||||||
|
|
||||||
|
* Fri Jan 28 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-22]
|
||||||
|
- Move /home/e-smith/* to e-smith-base. [MN00066635]
|
||||||
|
- Move generic-template expand to S05 position in sort, and
|
||||||
|
adjust-service to S90. Fix run-time error. [MN00066406]
|
||||||
|
- Don't attempt to execute non-executable action scripts.
|
||||||
|
|
||||||
|
* Thu Jan 27 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-21]
|
||||||
|
- Implicitly include template expand and services adjust actions
|
||||||
|
in each event if the respective metadata directory exists.
|
||||||
|
[MN00066406]
|
||||||
|
|
||||||
|
* Tue Jan 25 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-20]
|
||||||
|
- Add "adjust" to list of verbs which serviceControl groks,
|
||||||
|
to allow for "masq adjust". [MN00065576]
|
||||||
|
|
||||||
|
* Tue Jan 25 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-19]
|
||||||
|
- Add adjust-services generic action script [MN00065576]
|
||||||
|
|
||||||
|
* Tue Jan 18 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-18]
|
||||||
|
- Fix typo. [MN00064412]
|
||||||
|
- Improve die() message in esmith::config::STORE. [MN00064394]
|
||||||
|
|
||||||
|
* Mon Jan 17 2005 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-17]
|
||||||
|
- Fix broken logging (and reduce logging) in initialize_default_databases.
|
||||||
|
[MN00064412]
|
||||||
|
- Fix inappropriate use of global $_ in initialize_default_databases.
|
||||||
|
[MN00064415]
|
||||||
|
|
||||||
|
* Thu Dec 23 2004 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-16]
|
||||||
|
- Read templated file metadata (if any) from file tree under
|
||||||
|
/etc/e-smith/templates.metadata. Update expand-template to
|
||||||
|
use current API. [MN00061830]
|
||||||
|
|
||||||
|
* Tue Dec 14 2004 Mark Knox <markk@e-smith.com>
|
||||||
|
- [1.15.1-15]
|
||||||
|
- Change copyright date to 2004 [markk MN00060958]
|
||||||
|
|
||||||
|
* Fri Nov 5 2004 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-14]
|
||||||
|
- Fix Authen::PAM dependency header [charlieb MN00040240]
|
||||||
|
- Use kudzu's generated /etc/sysconfig/hwconf file for NIC detection
|
||||||
|
[charlieb MN00056220]
|
||||||
|
|
||||||
|
* Thu Oct 14 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-13]
|
||||||
|
- Updated esmith::ethernet's search code to remove File::Find, as it doesn't
|
||||||
|
get along with taint checking. [msoulier MN00052510]
|
||||||
|
|
||||||
|
* Wed Oct 13 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-12]
|
||||||
|
- Updated esmith::ethernet's search code such that is it more adaptable, and
|
||||||
|
recurses the directories given. [msoulier MN00052510]
|
||||||
|
|
||||||
|
* Wed Oct 13 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-11]
|
||||||
|
- Updated esmith::ethernet's search paths for network drivers.
|
||||||
|
[msoulier MN00052510]
|
||||||
|
|
||||||
|
* Mon Oct 4 2004 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-10]
|
||||||
|
- Remove dependency on perl(Filter::Handle) [charlieb MN00050075]
|
||||||
|
|
||||||
|
* Fri Sep 24 2004 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.1-09]
|
||||||
|
- Updated requires with new perl dependencies. [msoulier MN00040240]
|
||||||
|
- Remove "AutoReqProv: no" so that "Provides" headers are auto-generated.
|
||||||
|
[charlieb MN00040240]
|
||||||
|
- Remove anachronistic "require v5.6.0" directives. [charlieb MN00050370]
|
||||||
|
- Avoid use of Filter::Handle in esmith::template. [charlieb MN00050075]
|
||||||
|
|
||||||
|
* Fri Aug 27 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-08]
|
||||||
|
- Added esmith::util::network::isValidEmail function. [msoulier MN00023814]
|
||||||
|
|
||||||
|
* Thu Aug 26 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-07]
|
||||||
|
- Added svdisable to permissible actions in serviceControl.
|
||||||
|
[msoulier MN00043056]
|
||||||
|
|
||||||
|
* Tue Aug 10 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-06]
|
||||||
|
- Fixed new methods. Bad else case. [msoulier MN00044891]
|
||||||
|
|
||||||
|
* Fri Aug 6 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-05]
|
||||||
|
- Added a keys() method. [msoulier MN00041968]
|
||||||
|
|
||||||
|
* Fri Aug 6 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-04]
|
||||||
|
- Added set_prop and set_value methods in esmith::DB. [msoulier MN00044891]
|
||||||
|
|
||||||
|
* Tue Jul 20 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-03]
|
||||||
|
- Undeprecated esmith::util::serviceControl. [msoulier MN00043056]
|
||||||
|
|
||||||
|
* Fri Jun 25 2004 Tony Clayton <apc@e-smith.com>
|
||||||
|
- [1.15.1-02]
|
||||||
|
- Merge language_tag2locale() function from perl-I18N-LangTags [tonyc
|
||||||
|
MN00040170]
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.1-01]
|
||||||
|
- Rolling to collect patches.
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-23]
|
||||||
|
- Reordered the create code slightly to catch more errors.
|
||||||
|
[msoulier MN00035059]
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-22]
|
||||||
|
- Added yet more error handling and reporting. [msoulier MN00035059]
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-21]
|
||||||
|
- Fixed one $Error reference that I missed in the last rev.
|
||||||
|
[msoulier MN00035059]
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-20]
|
||||||
|
- Moved error handling from esmith::DB::db to esmith::DB, since it should not
|
||||||
|
be database implementation specific. [msoulier MN00035059]
|
||||||
|
|
||||||
|
* Fri May 28 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-19]
|
||||||
|
- Propagated migration failures up to calling code for reporting to syslog.
|
||||||
|
- Propagated creation failures up to calling code.
|
||||||
|
- Moved lexicon $Error in esmith::DB::db to a class property so it can be used
|
||||||
|
by subclasses. [msoulier MN00035059]
|
||||||
|
|
||||||
|
* Thu May 27 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-18]
|
||||||
|
- Changed print statements to calls to the logger. [msoulier MN00035059]
|
||||||
|
|
||||||
|
* Thu May 27 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-17]
|
||||||
|
- Added print statements to initialize-default-databases for post-install
|
||||||
|
debugging. [msoulier MN00035059]
|
||||||
|
|
||||||
|
* Fri May 7 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-16]
|
||||||
|
- Fixed migrate to report the caught error message. [msoulier MN00032503]
|
||||||
|
|
||||||
|
* Thu May 6 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-15]
|
||||||
|
- Added isValidHostname function to esmith::util::network.
|
||||||
|
[msoulier MN00024212]
|
||||||
|
|
||||||
|
* Fri Feb 13 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-14]
|
||||||
|
- Greatly simplified the _mysystem function by ripping out open3.
|
||||||
|
[msoulier dpar-20385]
|
||||||
|
|
||||||
|
* Fri Feb 13 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-13]
|
||||||
|
- Backed-out change in esmith::util, as it's non-trivial there with the exec.
|
||||||
|
Completed update of esmith::event, and separated the esmith::Logger class.
|
||||||
|
[msoulier dpar-20385]
|
||||||
|
|
||||||
|
* Fri Feb 13 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-12]
|
||||||
|
- Removed use of the logger entirely, replacing it with an abstracted
|
||||||
|
interface to the Sys::Syslog module. [msoulier dpar-20385]
|
||||||
|
|
||||||
|
* Thu Jan 8 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-11]
|
||||||
|
- Fixed esmith::util::network::isValidIP() so valid IP substrings no longer
|
||||||
|
return true values. [msoulier 9308]
|
||||||
|
|
||||||
|
* Thu Jan 8 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-10]
|
||||||
|
- Added a check in STORE in esmith::config for invalid attempts to set a value
|
||||||
|
without a type. Also escalated previous warnings for undef key and value to
|
||||||
|
fatal exceptions. [msoulier 7386]
|
||||||
|
|
||||||
|
* Thu Jan 8 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-09]
|
||||||
|
- Now trimming whitespace around keys and values during esmith::config STORE
|
||||||
|
events, to prevent invalid keys and values from being saved. [msoulier 7021]
|
||||||
|
|
||||||
|
* Mon Jan 5 2004 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-08]
|
||||||
|
- Fixed POD around merge_props. [msoulier 9482]
|
||||||
|
|
||||||
|
* Fri Nov 7 2003 Tony Clayton <apc@e-smith.com>
|
||||||
|
- [1.15.0-07]
|
||||||
|
- And again [tonyc 10569]
|
||||||
|
|
||||||
|
* Fri Nov 7 2003 Tony Clayton <apc@e-smith.com>
|
||||||
|
- [1.15.0-06]
|
||||||
|
- Change Merge API a bit, fix pod [tonyc 10569]
|
||||||
|
|
||||||
|
* Fri Nov 7 2003 Tony Clayton <apc@e-smith.com>
|
||||||
|
- [1.15.0-05]
|
||||||
|
- Add esmith::DB::Merge library [tonyc 10569]
|
||||||
|
|
||||||
|
* Fri Oct 10 2003 Michael Soulier <msoulier@e-smith.com>
|
||||||
|
- [1.15.0-04]
|
||||||
|
- Fixed AccountsDB.pm to handle group names with hyphens and periods, to match
|
||||||
|
the error message in the groups panel, and the rest of the group/user
|
||||||
|
behaviour. [msoulier 10236]
|
||||||
|
|
||||||
|
* Sun Sep 21 2003 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.0-03]
|
||||||
|
- Skip any directries when iterating over action scripts in esmith::event.
|
||||||
|
Add logfile timestamp symlinking to generic_template_expand action.
|
||||||
|
Use templates2expand and logfiles2timestamp subdirectories of the event
|
||||||
|
directory. Fix shebang line. [charlieb 10035]
|
||||||
|
|
||||||
|
* Thu Sep 18 2003 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.0-02]
|
||||||
|
- Add generic_template_expand action. [charlieb 10035]
|
||||||
|
|
||||||
|
* Thu Sep 18 2003 Charlie Brady <charlieb@e-smith.com>
|
||||||
|
- [1.15.0-01]
|
||||||
|
- Changing version to development stream number - 1.15.0
|
||||||
|
|
||||||
|
%prep
|
||||||
|
%setup
|
||||||
|
|
||||||
|
%pre
|
||||||
|
|
||||||
|
|
||||||
|
%post
|
||||||
|
|
||||||
|
%build
|
||||||
|
for event in post-install post-upgrade bootstrap-console-save console-save
|
||||||
|
do
|
||||||
|
mkdir -p root/etc/e-smith/events/$event
|
||||||
|
done
|
||||||
|
perl createlinks
|
||||||
|
find root/ -type f | xargs grep -l ____COPYYEARS____ | xargs sed -i -e 's/____COPYYEARS____/%{copykooz}/g'
|
||||||
|
|
||||||
|
%install
|
||||||
|
rm -rf $RPM_BUILD_ROOT
|
||||||
|
(cd root ; find . -depth -print | cpio -dump $RPM_BUILD_ROOT)
|
||||||
|
rm -f %{name}-%{version}-%{release}-filelist
|
||||||
|
/sbin/e-smith/genfilelist $RPM_BUILD_ROOT \
|
||||||
|
>%{name}-%{version}-%{release}-filelist
|
||||||
|
echo "%doc Copying" >> %{name}-%{version}-%{release}-filelist
|
||||||
|
echo "%doc Artistic" >> %{name}-%{version}-%{release}-filelist
|
||||||
|
echo "%doc LICENSE" >> %{name}-%{version}-%{release}-filelist
|
||||||
|
|
||||||
|
%clean
|
||||||
|
rm -rf $RPM_BUILD_ROOT
|
||||||
|
|
||||||
|
%files -f %{name}-%{version}-%{release}-filelist
|
||||||
|
%defattr(-,root,root)
|
3
root/etc/e-smith/db/configuration/migrate/00openRW
Normal file
3
root/etc/e-smith/db/configuration/migrate/00openRW
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
$DB = esmith::ConfigDB->open(${DB_FILENAME});
|
||||||
|
}
|
146
root/etc/e-smith/events/actions/adjust-services
Executable file
146
root/etc/e-smith/events/actions/adjust-services
Executable file
@ -0,0 +1,146 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
# copyright (C) 2005 Mitel Networks Corporation
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation; either version 2 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program; if not, write to the Free Software
|
||||||
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
#
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
package esmith;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use Errno;
|
||||||
|
use DirHandle;
|
||||||
|
|
||||||
|
my $event = shift || die "must give event name parameter";
|
||||||
|
chdir "/etc/e-smith/events/$event" or die "Couldn't chdir to event directory /etc/e-smith/events/$event: $!";
|
||||||
|
my $dh = DirHandle->new("services2adjust");
|
||||||
|
|
||||||
|
exit(0) unless $dh; # Nothing to do
|
||||||
|
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
use esmith::util;
|
||||||
|
|
||||||
|
my %param2char = (
|
||||||
|
down => 'd',
|
||||||
|
stop => 'd',
|
||||||
|
up => 'u',
|
||||||
|
start => 'u',
|
||||||
|
restart => 't',
|
||||||
|
sigterm => 't',
|
||||||
|
adjust => 'h',
|
||||||
|
reload => 'h',
|
||||||
|
sighup => 'h',
|
||||||
|
sigusr1 => '1',
|
||||||
|
sigusr2 => '2',
|
||||||
|
once => 'o',
|
||||||
|
pause => 'p',
|
||||||
|
alarm => 'a',
|
||||||
|
interrupt => 'i',
|
||||||
|
quit => 'q',
|
||||||
|
kill => 'k',
|
||||||
|
exit => 'x',
|
||||||
|
);
|
||||||
|
|
||||||
|
sub adjust_supervised_service
|
||||||
|
{
|
||||||
|
my ($s, @actions) = @_;
|
||||||
|
my $m = "control fifo for service $s: ";
|
||||||
|
unless (open(C, ">/service/$s/supervise/control"))
|
||||||
|
{
|
||||||
|
warn "Couldn't open $m$!";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
foreach my $p (@actions)
|
||||||
|
{
|
||||||
|
my $c = $param2char{$p};
|
||||||
|
unless ($c)
|
||||||
|
{
|
||||||
|
warn "Unrecognised param $p for service $s\n";
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
warn "adjusting supervised $s ($p)\n";
|
||||||
|
unless (print C $c)
|
||||||
|
{
|
||||||
|
warn "Couldn't write to $m$!";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
warn "Couldn't close $m$!" unless close(C);
|
||||||
|
}
|
||||||
|
|
||||||
|
my $conf = esmith::ConfigDB->open_ro || die "Couldn't open config db";
|
||||||
|
|
||||||
|
foreach my $service (grep { !/^\./ } $dh->read())
|
||||||
|
{
|
||||||
|
my $s = $conf->get($service);
|
||||||
|
unless ($s)
|
||||||
|
{
|
||||||
|
warn "No conf db entry for service $service\n";
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
my $f = "services2adjust/$service";
|
||||||
|
|
||||||
|
my @actions;
|
||||||
|
if (-l "$f")
|
||||||
|
{
|
||||||
|
@actions = ( readlink "$f" );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (open(F, $f))
|
||||||
|
{
|
||||||
|
# Read list of actions from the file, and untaint
|
||||||
|
@actions = map { chomp; /([a-z]+[12]?)/ ; $1 } <F>;
|
||||||
|
close(F);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
warn "Could not open $f: $!";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
# if service is supervised and not handled by systemd
|
||||||
|
if (-d "/service/$service" && glob("/etc/rc7.d/S??$service"))
|
||||||
|
{
|
||||||
|
my $enabled = ($s->prop('status') || 'disabled') eq 'enabled';
|
||||||
|
adjust_supervised_service($service,
|
||||||
|
# stop the service if it is now disabled
|
||||||
|
$enabled ? () : 'down',
|
||||||
|
# Send the specified signal(s) to the running daemon
|
||||||
|
@actions,
|
||||||
|
# bring the service up if it is enabled (and we're not
|
||||||
|
# stopping it or running it once)
|
||||||
|
($enabled && !grep { /^(down|stop|d|once|o)$/ } @actions) ? 'up' : (),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
# for service handled by former sysvinit or directly with systemd
|
||||||
|
else
|
||||||
|
{
|
||||||
|
my $enabled = ($s->prop('status') || 'disabled') eq 'enabled';
|
||||||
|
# bring the service up if it is enabled (and we're not stopping it or running it once, or using signal able to start it)
|
||||||
|
unshift(@actions,'start') if ($enabled && !grep { /^(down|stop|d|once|o|start|restart|reload-or-restart)$/ } @actions) ;
|
||||||
|
# stop the service if it is disabled
|
||||||
|
@actions = ('stop') unless $enabled;
|
||||||
|
foreach (@actions)
|
||||||
|
{
|
||||||
|
warn "adjusting non-supervised $service ($_)\n";
|
||||||
|
esmith::util::serviceControl(
|
||||||
|
NAME => $service,
|
||||||
|
ACTION => $_,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
61
root/etc/e-smith/events/actions/generic_template_expand
Normal file
61
root/etc/e-smith/events/actions/generic_template_expand
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
#! /usr/bin/perl -w
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
# copyright (C) 2003-2007 Mitel Networks Corporation
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation; either version 2 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program; if not, write to the Free Software
|
||||||
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
#
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use File::Find;
|
||||||
|
use File::Copy;
|
||||||
|
use esmith::templates;
|
||||||
|
|
||||||
|
sub expand;
|
||||||
|
|
||||||
|
my $event = shift or die "Event name is required\n";
|
||||||
|
|
||||||
|
my $filename;
|
||||||
|
my %args;
|
||||||
|
my ($param, $value);
|
||||||
|
my $templates_dir = "/etc/e-smith/events/$event/templates2expand";
|
||||||
|
exit 0 unless -d $templates_dir;
|
||||||
|
|
||||||
|
chdir $templates_dir or die "Could not chdir to $templates_dir: $!\n";;
|
||||||
|
# Walk the tree and expand all templates referenced thereunder.
|
||||||
|
find({
|
||||||
|
no_chdir => 1,
|
||||||
|
follow => 0,
|
||||||
|
wanted => \&expand,
|
||||||
|
},
|
||||||
|
'.'
|
||||||
|
);
|
||||||
|
|
||||||
|
exit 0;
|
||||||
|
|
||||||
|
sub expand
|
||||||
|
{
|
||||||
|
return unless -f $_;
|
||||||
|
# For each file found, read the file to find
|
||||||
|
# processTemplate args, then expand the template
|
||||||
|
s/^\.//;
|
||||||
|
$filename = $_;
|
||||||
|
warn "expanding $filename\n";
|
||||||
|
esmith::templates::processTemplate({
|
||||||
|
MORE_DATA => { EVENT => $event },
|
||||||
|
TEMPLATE_PATH => $filename,
|
||||||
|
OUTPUT_FILENAME => $filename,
|
||||||
|
});
|
||||||
|
}
|
35
root/etc/e-smith/events/actions/initialize-default-databases
Normal file
35
root/etc/e-smith/events/actions/initialize-default-databases
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
# copyright (C) 1999-2003 Mitel Networks Corporation
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation; either version 2 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program; if not, write to the Free Software
|
||||||
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
#
|
||||||
|
# Technical support for this program is available from Mitel Networks
|
||||||
|
# Please visit our web site www.mitel.com/sme/ for details.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
package esmith;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use esmith::util;
|
||||||
|
|
||||||
|
if (esmith::util::initialize_default_databases())
|
||||||
|
{
|
||||||
|
exit 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
exit 1;
|
||||||
|
}
|
0
root/etc/e-smith/events/e-smith-lib-update/.gitignore
vendored
Normal file
0
root/etc/e-smith/events/e-smith-lib-update/.gitignore
vendored
Normal file
10
root/etc/e-smith/templates-default/template-begin
Normal file
10
root/etc/e-smith/templates-default/template-begin
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
#------------------------------------------------------------
|
||||||
|
# !!DO NOT MODIFY THIS FILE!!
|
||||||
|
#
|
||||||
|
# Manual changes will be lost when this file is regenerated.
|
||||||
|
#
|
||||||
|
# Please read the developer's guide, which is available
|
||||||
|
# at http://www.contribs.org/development/
|
||||||
|
#
|
||||||
|
# Copyright (C) 1999-2006 Mitel Networks Corporation
|
||||||
|
#------------------------------------------------------------
|
13
root/etc/e-smith/templates-default/template-begin-html
Normal file
13
root/etc/e-smith/templates-default/template-begin-html
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
{
|
||||||
|
$OUT = <<HERE;
|
||||||
|
<!--
|
||||||
|
HERE
|
||||||
|
|
||||||
|
$OUT .=
|
||||||
|
Text::Template::_load_text("/etc/e-smith/templates-default/template-begin");
|
||||||
|
|
||||||
|
$OUT .= <<HERE;
|
||||||
|
-->
|
||||||
|
HERE
|
||||||
|
|
||||||
|
}
|
8
root/etc/e-smith/templates-default/template-begin-pam
Normal file
8
root/etc/e-smith/templates-default/template-begin-pam
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
$OUT = <<HERE;
|
||||||
|
#%PAM-1.0
|
||||||
|
HERE
|
||||||
|
|
||||||
|
$OUT .=
|
||||||
|
Text::Template::_load_text("/etc/e-smith/templates-default/template-begin");
|
||||||
|
}
|
8
root/etc/e-smith/templates-default/template-begin-perl
Normal file
8
root/etc/e-smith/templates-default/template-begin-perl
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
$OUT = <<HERE;
|
||||||
|
#!/usr/bin/perl -w
|
||||||
|
HERE
|
||||||
|
|
||||||
|
$OUT .=
|
||||||
|
Text::Template::_load_text("/etc/e-smith/templates-default/template-begin");
|
||||||
|
}
|
14
root/etc/e-smith/templates-default/template-begin-php
Normal file
14
root/etc/e-smith/templates-default/template-begin-php
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
{
|
||||||
|
$OUT = <<HERE;
|
||||||
|
<?php
|
||||||
|
/*
|
||||||
|
HERE
|
||||||
|
|
||||||
|
$OUT .=
|
||||||
|
Text::Template::_load_text("/etc/e-smith/templates-default/template-begin");
|
||||||
|
|
||||||
|
$OUT .= <<HERE;
|
||||||
|
*/
|
||||||
|
HERE
|
||||||
|
|
||||||
|
}
|
8
root/etc/e-smith/templates-default/template-begin-shell
Normal file
8
root/etc/e-smith/templates-default/template-begin-shell
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
$OUT = <<HERE;
|
||||||
|
#!/bin/sh
|
||||||
|
HERE
|
||||||
|
|
||||||
|
$OUT .=
|
||||||
|
Text::Template::_load_text("/etc/e-smith/templates-default/template-begin");
|
||||||
|
}
|
1
root/etc/e-smith/templates-default/template-end-php
Normal file
1
root/etc/e-smith/templates-default/template-end-php
Normal file
@ -0,0 +1 @@
|
|||||||
|
?>
|
81
root/etc/e-smith/tests/10e-smith-lib/accounts.conf
Normal file
81
root/etc/e-smith/tests/10e-smith-lib/accounts.conf
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
# DO NOT MODIFY THIS FILE.
|
||||||
|
# This file is automatically maintained by the Mitel Networks SME Server
|
||||||
|
# configuration software. Manually editing this file may put your
|
||||||
|
# system in an unknown state.
|
||||||
|
#
|
||||||
|
# updated: Thu Mar 28 15:29:31 2002
|
||||||
|
Bart.Simpson=pseudonym|Account|bart
|
||||||
|
Bart_Simpson=pseudonym|Account|bart
|
||||||
|
Global=system
|
||||||
|
Primary=system
|
||||||
|
adm=system|Gid|4|Uid|3
|
||||||
|
admin=system|Gid|101|Uid|101
|
||||||
|
alias=system|Gid|400|Uid|400
|
||||||
|
apache=existing|Gid|48|Uid|48
|
||||||
|
bart=user|FirstName|Bart|LastName|Simpson
|
||||||
|
bin=system|Gid|1|Uid|1
|
||||||
|
cdrom=system
|
||||||
|
cgi-bin=url
|
||||||
|
console=system
|
||||||
|
daemon=system|Gid|2|Uid|2
|
||||||
|
dip=system|Gid|40
|
||||||
|
disk=system|Gid|6
|
||||||
|
dns=existing|Gid|53|Uid|53
|
||||||
|
e-smith-manager=url
|
||||||
|
e-smith-password=url
|
||||||
|
everyone=pseudonym|Account|shared|Visible|internal
|
||||||
|
floppy=system|Gid|19
|
||||||
|
ftp=system|Gid|50|Uid|14
|
||||||
|
games=system|Gid|20|Uid|12
|
||||||
|
global=system
|
||||||
|
gopher=system|Gid|30|Uid|13
|
||||||
|
halt=system
|
||||||
|
homes=system
|
||||||
|
kmem=system|Gid|9
|
||||||
|
ldap=existing|Gid|55|Uid|55
|
||||||
|
lp=system|Gid|7|Uid|4
|
||||||
|
mail=system|Gid|12|Uid|8
|
||||||
|
mailer-daemon=pseudonym|Account|admin
|
||||||
|
man=system|Gid|15
|
||||||
|
mem=system|Gid|8
|
||||||
|
mysql=existing|Gid|27|Uid|27
|
||||||
|
named=existing|Gid|25|Uid|25
|
||||||
|
netlogon=netlogon|Comment|placeholder for netlogon share
|
||||||
|
news=system|Gid|13|Uid|9
|
||||||
|
nofiles=system|Gid|400
|
||||||
|
operator=system|Gid|0|Uid|11
|
||||||
|
postgres=system
|
||||||
|
postmaster=pseudonym|Account|admin
|
||||||
|
primary=system
|
||||||
|
printers=system
|
||||||
|
public=system|Gid|103|Uid|102
|
||||||
|
qmail=system|Gid|401
|
||||||
|
qmaild=system|Gid|400|Uid|401
|
||||||
|
qmaill=system|Gid|400|Uid|402
|
||||||
|
qmailp=system|Gid|400|Uid|403
|
||||||
|
qmailq=system|Gid|401|Uid|404
|
||||||
|
qmailr=system|Gid|401|Uid|405
|
||||||
|
qmails=system|Gid|401|Uid|406
|
||||||
|
qmailscan=existing|Gid|407|Uid|407
|
||||||
|
root=system|Gid|0|Uid|0
|
||||||
|
schwern=user|Uid|500|Gid|501|FirstName|Michael|LastName|Schwern
|
||||||
|
server-manager=url
|
||||||
|
server-manual=url
|
||||||
|
shared=system|Gid|500|Visible|internal
|
||||||
|
shutdown=system
|
||||||
|
simpsons=group|Description|bar|Gid|5005|Members|bart,lisa,homer,maggie|Uid|5005
|
||||||
|
slocate=system
|
||||||
|
somegroup=group|Gid|42|Members|admin
|
||||||
|
squid=system|Gid|23|Uid|23
|
||||||
|
sync=system
|
||||||
|
sys=system|Gid|3
|
||||||
|
trend=existing|Gid|408|Uid|408
|
||||||
|
tty=system|Gid|5
|
||||||
|
user-password=url
|
||||||
|
users=system|Gid|100
|
||||||
|
utmp=system|Gid|22
|
||||||
|
uucp=system|Gid|14|Uid|10
|
||||||
|
webmail=url
|
||||||
|
wheel=system|Gid|10
|
||||||
|
www=system|Gid|102|Uid|100
|
||||||
|
wwwpublic=system
|
95
root/etc/e-smith/tests/10e-smith-lib/config.t
Normal file
95
root/etc/e-smith/tests/10e-smith-lib/config.t
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
|
||||||
|
# Overall tests for esmith::config
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use File::Copy;
|
||||||
|
|
||||||
|
use Test::More 'no_plan';
|
||||||
|
use_ok('esmith::config');
|
||||||
|
|
||||||
|
my %Expect = ( foo => 'bar',
|
||||||
|
'this key' => 'has whitespace',
|
||||||
|
'that key ' => 'has trailing whitespace',
|
||||||
|
' another key' => 'has leading whitespace',
|
||||||
|
'this value' => ' has leading whitespace',
|
||||||
|
'that value' => 'has trailing whitespace ',
|
||||||
|
'tricky value' => 'with=equals.',
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
# so we don't bork the original.
|
||||||
|
my $Scratch = '10e-smith-lib/mydummy.conf';
|
||||||
|
copy('10e-smith-lib/dummy.conf', $Scratch);
|
||||||
|
END { unlink $Scratch }
|
||||||
|
|
||||||
|
my %config;
|
||||||
|
tie %config, 'esmith::config', $Scratch;
|
||||||
|
ok( tied %config, 'tie worked' );
|
||||||
|
is_deeply( \%config, \%Expect, 'read in the config properly' );
|
||||||
|
|
||||||
|
# Test the tied interface.
|
||||||
|
is_deeply( [sort keys %config], [sort keys %Expect], 'keys' );
|
||||||
|
is_deeply( [sort values %config], [sort values %Expect], 'values' );
|
||||||
|
|
||||||
|
is_deeply( [@config{'foo', 'this key'}], [@Expect{'foo', 'this key'}],
|
||||||
|
'hash slice');
|
||||||
|
|
||||||
|
$config{foo} = 'baz';
|
||||||
|
is( $config{foo}, 'baz', 'STORE' );
|
||||||
|
|
||||||
|
my %config_copy;
|
||||||
|
tie %config_copy, 'esmith::config', $Scratch;
|
||||||
|
is( $config_copy{foo}, 'baz', ' STORE saved' );
|
||||||
|
|
||||||
|
SKIP: {
|
||||||
|
skip "Locking is broken in perl 5.6.0", 2 if $] eq 5.006;
|
||||||
|
|
||||||
|
tied(%config_copy)->_lock_write;
|
||||||
|
my $start_time = time;
|
||||||
|
{
|
||||||
|
local $ENV{PERL5LIB} = join ':', @INC;
|
||||||
|
system(qq{$^X -Mesmith::config -e 'alarm 4; tie %config, "esmith::config", q{$Scratch}'});
|
||||||
|
}
|
||||||
|
|
||||||
|
cmp_ok( time - 2, '<=', $start_time, 'write locks dont prevent read' );
|
||||||
|
|
||||||
|
|
||||||
|
tied(%config_copy)->_lock_write;
|
||||||
|
$start_time = time;
|
||||||
|
{
|
||||||
|
local $ENV{PERL5LIB} = join ':', @INC;
|
||||||
|
system(qq{$^X -Mesmith::config -e 'alarm 4; tie %config, "esmith::config", q{$Scratch}; \$config{foo} = 42'});
|
||||||
|
}
|
||||||
|
|
||||||
|
cmp_ok( time - 1, '>=', $start_time, 'write locks prevent writes' );
|
||||||
|
};
|
||||||
|
|
||||||
|
my $new_conf = 'I_dont_exist';
|
||||||
|
unlink $new_conf;
|
||||||
|
ok( !-e $new_conf, 'config file doesnt exist' );
|
||||||
|
END { unlink $new_conf }
|
||||||
|
|
||||||
|
tie %config, 'esmith::config', $new_conf;
|
||||||
|
is( keys %config, 0, 'new() from nonexistent config' );
|
||||||
|
$config{wibble} = 'wobble';
|
||||||
|
|
||||||
|
tie %config_copy, 'esmith::config', $new_conf;
|
||||||
|
is( $config_copy{wibble}, 'wobble', ' new config file written' );
|
||||||
|
|
||||||
|
|
||||||
|
SKIP: {
|
||||||
|
skip "Locking is broken in perl 5.6.0", 1 if $] eq 5.006;
|
||||||
|
|
||||||
|
# There was a bug where if you set something to its existing value
|
||||||
|
# it wouldn't unlock properly.
|
||||||
|
my $Alarm;
|
||||||
|
eval {
|
||||||
|
local $SIG{ALRM} = sub { $Alarm = 1; die "ALARM!\n"; };
|
||||||
|
alarm 1;
|
||||||
|
$config_copy{wibble} = $config_copy{wibble};
|
||||||
|
$config{wibble} = 42;
|
||||||
|
alarm 0;
|
||||||
|
};
|
||||||
|
ok( !$Alarm, 'Unlocking works for setting the same value' );
|
||||||
|
};
|
13
root/etc/e-smith/tests/10e-smith-lib/config_taint.t
Normal file
13
root/etc/e-smith/tests/10e-smith-lib/config_taint.t
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
#!/usr/bin/perl -Tw
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use esmith::config;
|
||||||
|
use Test::More tests => 2;
|
||||||
|
|
||||||
|
my %config;
|
||||||
|
tie %config, 'esmith::config', '10e-smith-lib/dummy.conf';
|
||||||
|
my $value = $config{foo};
|
||||||
|
|
||||||
|
# Config values *should* be tained, but code depends on them not being.
|
||||||
|
is( $value, 'bar', 'tied to the dummy database' );
|
||||||
|
ok( eval { () = join('', $value), kill 0; 1; }, 'config values not tainted' );
|
103
root/etc/e-smith/tests/10e-smith-lib/configuration.conf
Normal file
103
root/etc/e-smith/tests/10e-smith-lib/configuration.conf
Normal file
@ -0,0 +1,103 @@
|
|||||||
|
# DO NOT MODIFY THIS FILE.
|
||||||
|
# This file is automatically maintained by the Mitel Networks SME Server
|
||||||
|
# configuration software. Manually editing this file may put your
|
||||||
|
# system in an unknown state.
|
||||||
|
#
|
||||||
|
# updated: Fri Feb 28 16:41:34 2003
|
||||||
|
AccessType=dedicated
|
||||||
|
ActiveAccounts=0
|
||||||
|
AdminEmail=
|
||||||
|
ConsoleMode=login
|
||||||
|
ContactEmail=
|
||||||
|
ContactName=
|
||||||
|
ContactOrg=
|
||||||
|
DialupConnOffice=long
|
||||||
|
DialupConnOutside=long
|
||||||
|
DialupConnWeekend=long
|
||||||
|
DialupFreqOffice=every15min
|
||||||
|
DialupFreqOutside=everyhour
|
||||||
|
DialupFreqWeekend=everyhour
|
||||||
|
DialupModemDevice=/dev/ttyS1
|
||||||
|
DialupPhoneNumber=
|
||||||
|
DialupUserAccount=useraccount
|
||||||
|
DialupUserPassword=userpassword
|
||||||
|
DomainName=e-smith.com
|
||||||
|
DynDnsAccount=dnsaccount
|
||||||
|
DynDnsPassword=dnspassword
|
||||||
|
DynDnsService=off
|
||||||
|
EmailUnknownUser=return
|
||||||
|
EthernetDriver1=pcnet32
|
||||||
|
EthernetDriver2=unknown
|
||||||
|
ExternalDHCP=off
|
||||||
|
ExternalNetmask=255.255.255.0
|
||||||
|
GatewayIP=192.168.16.1
|
||||||
|
LocalIP=192.168.16.228
|
||||||
|
LocalNetmask=255.255.255.0
|
||||||
|
MinUid=5000
|
||||||
|
PasswordSet=yes
|
||||||
|
PreviousConfiguration=/home/e-smith/db/configuration.previous
|
||||||
|
SMTPSmartHost=
|
||||||
|
SambaDomainMaster=no
|
||||||
|
SambaServerName=pretz
|
||||||
|
SambaWorkgroup=mitel-networks
|
||||||
|
ServiceAccountId=
|
||||||
|
ServiceDomainName=
|
||||||
|
ServiceTargetIP=
|
||||||
|
SquidParent=
|
||||||
|
SquidParentPort=
|
||||||
|
StatusReports=off
|
||||||
|
SystemMode=serveronly
|
||||||
|
SystemName=pretz
|
||||||
|
TimeZone=US/Eastern
|
||||||
|
UnsavedChanges=yes
|
||||||
|
atalk=service|InitscriptOrder|91|status|enabled
|
||||||
|
auth=service|access|public|status|enabled
|
||||||
|
bazbar=service|status|enabled
|
||||||
|
blades=service|Host|service.e-smith.com|status|enabled
|
||||||
|
bootstrap-console=service|InitscriptOrder|35|Run|no|status|enabled
|
||||||
|
branding=service|modified|000000000000|status|enabled
|
||||||
|
crond=service|InitscriptOrder|40|status|enabled
|
||||||
|
ctrlaltdel=service|status|enabled
|
||||||
|
dhcpd=service|InitscriptOrder|65|end|192.168.16.250|start|192.168.16.65|status|disabled
|
||||||
|
diald=service|InitscriptOrder|57|status|disabled
|
||||||
|
fetchmail=service|FreqOffice|every5min|FreqOutside|every30min|FreqWeekend|never|Method|standard|SecondaryMailAccount|popaccount|SecondaryMailPassword|poppassword|SecondaryMailServer|mail.myisp.xxx|status|disabled
|
||||||
|
flexbackup=backupservice|erase_rewind_only|true
|
||||||
|
ftp=service|access|private|accessLimits|off|status|enabled
|
||||||
|
hdparm=service|InitscriptOrder|40|status|disabled
|
||||||
|
horde=service|status|disabled
|
||||||
|
httpd-admin=service|InitscriptOrder|86|status|enabled
|
||||||
|
httpd-e-smith=service|InitscriptOrder|85|access|private|status|enabled
|
||||||
|
imap=service|access|private|status|enabled
|
||||||
|
imp=service|status|disabled
|
||||||
|
ippp=service|InitscriptOrder|55|status|enabled
|
||||||
|
ipsec=service|InitscriptOrder|90|PubKey|0sAQOoIKaOMuDqSdCZJXgv9QI86DAuAwbbvn8uoKn2lRQ9ZVPTn9Ow5znhuw/GopsYD2eujhtvkQo7fszAhWbEpn+lW2LzLCbZYaDov7j8Q9CpeJSVgeuzaBcw3OenSL3ltTwWWtG0pvyaYsfepNqVYvo64YVmrxo0O7dCECySMVBZkQ==|status|disabled
|
||||||
|
isdn=service|Protocol|2|UseSyncPPP|yes|status|disabled
|
||||||
|
keytable=service|InitscriptOrder|25|status|enabled
|
||||||
|
ldap=service|InitscriptOrder|80|access|private|defaultCity|Ottawa|defaultCompany|XYZ Corporation|defaultDepartment|Main|defaultPhoneNumber|555-5555|defaultStreet|123 Main Street|status|enabled
|
||||||
|
lilo=service|AddressMode|linear
|
||||||
|
local=service|InitscriptOrder|99|status|enabled
|
||||||
|
lpd=service|InitscriptOrder|60|status|enabled
|
||||||
|
mariadb=service|InitscriptOrder|90|status|enabled
|
||||||
|
masq=service|InitscriptOrder|06|Logging|none|Stealth|no|status|disabled
|
||||||
|
modSSL=service|status|enabled
|
||||||
|
mysql.init=service|InitscriptOrder|99|status|enabled
|
||||||
|
named=service|chroot|yes|status|enabled|Forwarder1|1.2.3.4
|
||||||
|
network=service|InitscriptOrder|10|status|enabled
|
||||||
|
ntpd=service|InitscriptOrder|55|status|disabled
|
||||||
|
php=service|status|enabled
|
||||||
|
popd=service|access|private|status|enabled
|
||||||
|
pppoe=service|DemandIdleTime|no|InitscriptOrder|57|SynchronousPPP|no|status|disabled
|
||||||
|
pptpd=service|sessions|10|status|disabled
|
||||||
|
qmail=service|InitscriptOrder|80|status|enabled
|
||||||
|
random=service|InitscriptOrder|20|status|enabled
|
||||||
|
rsyslog=service|InitscriptOrder|05|status|enabled
|
||||||
|
scanner=service|ScannerFns|iscan|UpdateTime|1:14|scanMail|yes|status|enabled
|
||||||
|
smb=service|InitscriptOrder|91|RoamingProfiles|no|status|enabled
|
||||||
|
smtpd=service|access|public|status|enabled
|
||||||
|
smtpfwdd=service|InitscriptOrder|81|status|enabled
|
||||||
|
squid=service|InitscriptOrder|90|status|enabled
|
||||||
|
sshd=service|InitscriptOrder|85|PasswordAuthentication|yes|PermitRootLogin|yes|access|private|status|enabled
|
||||||
|
sync=service|Host|service.e-smith.com|LastId|0|SuccessId|0|SyncFrequency|1|SyncMinute|57|status|disabled
|
||||||
|
telnet=service|access|private|status|disabled
|
||||||
|
wibble=42
|
||||||
|
xinetd=service|InitscriptOrder|50|status|enabled
|
215
root/etc/e-smith/tests/10e-smith-lib/db.t
Normal file
215
root/etc/e-smith/tests/10e-smith-lib/db.t
Normal file
@ -0,0 +1,215 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
|
||||||
|
# Overall test for esmith::db
|
||||||
|
|
||||||
|
use File::Copy;
|
||||||
|
use esmith::TestUtils;
|
||||||
|
use Test::More 'no_plan';
|
||||||
|
use_ok('esmith::db');
|
||||||
|
|
||||||
|
my %Expect = (
|
||||||
|
Foo => ['Bar', {}],
|
||||||
|
Night => ['Day', {}],
|
||||||
|
Squid => ['cephalopod', {
|
||||||
|
arms => 10,
|
||||||
|
species => 'Loligo',
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
# Ensure that empty values are read in properly.
|
||||||
|
Octopus => ['cephalopod', {
|
||||||
|
arms => 8,
|
||||||
|
species => '',
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
# Ensure that escaped pipes are read in properly.
|
||||||
|
Pipe => ['art', { pipe => 'this is not a \| got that?'}],
|
||||||
|
|
||||||
|
# Ensure that escaped newlines are handled properly.
|
||||||
|
Haiku => ['poem', { words =>
|
||||||
|
"Damian Conway\n".
|
||||||
|
"God damn! Damian Conway\n".
|
||||||
|
"Damian Conway"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
my $Scratch_Conf = '10e-smith-lib/db_scratch.conf';
|
||||||
|
copy '10e-smith-lib/db_dummy.conf', $Scratch_Conf;
|
||||||
|
END { unlink $Scratch_Conf }
|
||||||
|
|
||||||
|
my %config;
|
||||||
|
tie %config, 'esmith::config', $Scratch_Conf;
|
||||||
|
ok( tied %config, 'tied to the dummy config file' );
|
||||||
|
isnt( keys %config, 0, ' and theres something in there' );
|
||||||
|
|
||||||
|
is( db_get_type(\%config, 'Foo'), 'Bar', 'simple db_get_type' );
|
||||||
|
|
||||||
|
my @keys = db_get(\%config);
|
||||||
|
is_deeply( [sort @keys], [sort keys %Expect],
|
||||||
|
'db_get() all keys' );
|
||||||
|
|
||||||
|
|
||||||
|
foreach my $key (@keys) {
|
||||||
|
my($type, %properties) = db_get(\%config, $key);
|
||||||
|
my($exp_type, $exp_properties) = @{$Expect{$key}};
|
||||||
|
|
||||||
|
is( $type, $exp_type, "db_get - type - $key" );
|
||||||
|
is( db_get_type(\%config, $key), $exp_type, "db_get_type" );
|
||||||
|
is_deeply( \%properties, $exp_properties, "db_get - prop" );
|
||||||
|
is_deeply( {db_get_prop(\%config, $key)}, $exp_properties,
|
||||||
|
"db_get_prop - all properties");
|
||||||
|
while( my($prop, $val) = each %properties ) {
|
||||||
|
is( db_get_prop(\%config, $key, $prop), $val,
|
||||||
|
"db_get_prop - single prop - $prop");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
is( db_get_type(\%config, 'I_dont_exist'), undef,
|
||||||
|
'db_get_type on non-existent key' );
|
||||||
|
is( db_get_prop(\%config, 'I_dont_exist'), undef,
|
||||||
|
' db_get_prop' );
|
||||||
|
is( db_get_prop(\%config, 'Squid', 'feet'), undef,
|
||||||
|
'db_get_prop on non-existent prop' );
|
||||||
|
is( db_delete_prop(\%config, 'I_dont_exist', 'feet'), undef,
|
||||||
|
'db_delete_prop on non-existent key' );
|
||||||
|
|
||||||
|
is( db_get(\%config, 'Squid'), 'cephalopod|arms|10|species|Loligo',
|
||||||
|
'db_get a raw value');
|
||||||
|
|
||||||
|
{
|
||||||
|
package TieOut;
|
||||||
|
|
||||||
|
sub TIEHANDLE {
|
||||||
|
bless( \(my $scalar), $_[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub PRINT {
|
||||||
|
my $self = shift;
|
||||||
|
$$self .= join('', @_);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub read {
|
||||||
|
my $self = shift;
|
||||||
|
return substr($$self, 0, length($$self), '');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
my $out = tie *STDOUT, 'TieOut';
|
||||||
|
db_show(\%config);
|
||||||
|
is( $out->read, <<SHOW, 'db_show() all' );
|
||||||
|
Foo=Bar
|
||||||
|
Haiku=poem
|
||||||
|
words=Damian Conway\nGod damn! Damian Conway\nDamian Conway
|
||||||
|
Night=Day
|
||||||
|
Octopus=cephalopod
|
||||||
|
arms=8
|
||||||
|
species=
|
||||||
|
Pipe=art
|
||||||
|
pipe=this is not a \\| got that?
|
||||||
|
Squid=cephalopod
|
||||||
|
arms=10
|
||||||
|
species=Loligo
|
||||||
|
SHOW
|
||||||
|
|
||||||
|
db_show(\%config, 'Squid');
|
||||||
|
is( $out->read, <<SHOW, 'db_show() one key' );
|
||||||
|
Squid=cephalopod
|
||||||
|
arms=10
|
||||||
|
species=Loligo
|
||||||
|
SHOW
|
||||||
|
|
||||||
|
|
||||||
|
db_print(\%config);
|
||||||
|
is( $out->read, <<PRINT, 'db_print all' );
|
||||||
|
Foo=Bar
|
||||||
|
Haiku=poem|words|Damian Conway\nGod damn! Damian Conway\nDamian Conway
|
||||||
|
Night=Day
|
||||||
|
Octopus=cephalopod|arms|8|species|
|
||||||
|
Pipe=art|pipe|this is not a \\| got that?
|
||||||
|
Squid=cephalopod|arms|10|species|Loligo
|
||||||
|
PRINT
|
||||||
|
|
||||||
|
|
||||||
|
db_print(\%config, 'Squid');
|
||||||
|
is( $out->read, <<PRINT, 'db_print one key' );
|
||||||
|
Squid=cephalopod|arms|10|species|Loligo
|
||||||
|
PRINT
|
||||||
|
|
||||||
|
|
||||||
|
db_print_type(\%config);
|
||||||
|
is( $out->read, <<PRINT_TYPE, 'db_print_type all keys' );
|
||||||
|
Foo=Bar
|
||||||
|
Haiku=poem
|
||||||
|
Night=Day
|
||||||
|
Octopus=cephalopod
|
||||||
|
Pipe=art
|
||||||
|
Squid=cephalopod
|
||||||
|
PRINT_TYPE
|
||||||
|
|
||||||
|
db_print_type(\%config, 'Squid');
|
||||||
|
is( $out->read, <<PRINT_TYPE, 'db_print_type one key' );
|
||||||
|
Squid=cephalopod
|
||||||
|
PRINT_TYPE
|
||||||
|
|
||||||
|
db_print_prop(\%config, 'Squid');
|
||||||
|
is( $out->read, <<PRINT_PROP, 'db_print_prop all props' );
|
||||||
|
arms=10
|
||||||
|
species=Loligo
|
||||||
|
PRINT_PROP
|
||||||
|
|
||||||
|
db_print_prop(\%config, 'Squid', 'arms');
|
||||||
|
is( $out->read, <<PRINT_PROP, 'db_print_prop one prop' );
|
||||||
|
arms=10
|
||||||
|
PRINT_PROP
|
||||||
|
|
||||||
|
undef $out;
|
||||||
|
untie *STDOUT;
|
||||||
|
|
||||||
|
db_set(\%config, 'Wibble', 'dribble|hip|hop');
|
||||||
|
my($type, %props) = db_get(\%config, 'Wibble');
|
||||||
|
is( $type, 'dribble', 'db_set with raw value' );
|
||||||
|
is_deeply( \%props, { hip => 'hop' }, ' again' );
|
||||||
|
|
||||||
|
db_set(\%config, 'Wibble', 'word', { thingy => 'yep' });
|
||||||
|
($type, %props) = db_get(\%config, 'Wibble');
|
||||||
|
is( $type, 'word', 'db_set');
|
||||||
|
is_deeply( \%props, { thingy => 'yep' } );
|
||||||
|
|
||||||
|
db_set_type(\%config, 'Wibble', 'yibble');
|
||||||
|
is( db_get_type(\%config, 'Wibble'), 'yibble', 'db_set_type' );
|
||||||
|
|
||||||
|
db_set_prop(\%config, 'Wibble', har => 'far');
|
||||||
|
is( db_get_prop(\%config, 'Wibble', 'har'), 'far', 'db_set_prop' );
|
||||||
|
|
||||||
|
|
||||||
|
### Test read-only open()
|
||||||
|
my $scratch = scratch_copy('10e-smith-lib/configuration.conf');
|
||||||
|
ok( chmod 0444, $scratch );
|
||||||
|
my $ro_db = esmith::DB::db->open_ro($scratch);
|
||||||
|
|
||||||
|
eval { $ro_db->new_record('wibble', { type => 'something' }) };
|
||||||
|
like( $@, qr/read-only/ );
|
||||||
|
|
||||||
|
my $sshd = $ro_db->get('sshd');
|
||||||
|
eval { $sshd->set_prop('foo', 'bar') };
|
||||||
|
like( $@, qr/read-only/ );
|
||||||
|
is( $sshd->prop('foo'), undef );
|
||||||
|
|
||||||
|
eval { $sshd->delete_prop('status') };
|
||||||
|
like( $@, qr/read-only/ );
|
||||||
|
isnt( $sshd->prop('status'), undef );
|
||||||
|
|
||||||
|
eval { $sshd->merge_props(foo => 'bar') };
|
||||||
|
like( $@, qr/read-only/ );
|
||||||
|
is( $sshd->prop('foo'), undef );
|
||||||
|
|
||||||
|
eval { $sshd->reset_props(foo => 'bar') };
|
||||||
|
like( $@, qr/read-only/ );
|
||||||
|
is( $sshd->prop('foo'), undef );
|
||||||
|
|
||||||
|
eval { $sshd->delete };
|
||||||
|
like( $@, qr/read-only/ );
|
||||||
|
ok( $ro_db->get('sshd') );
|
@ -0,0 +1 @@
|
|||||||
|
invalid
|
@ -0,0 +1 @@
|
|||||||
|
enabled
|
@ -0,0 +1 @@
|
|||||||
|
service
|
@ -0,0 +1 @@
|
|||||||
|
enabled
|
@ -0,0 +1,4 @@
|
|||||||
|
{
|
||||||
|
$DB = esmith::ConfigDB->open("${DB_FILENAME}");
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
$DB->new_record("quux", {type=>'service', status=>'enabled'});
|
||||||
|
}
|
7
root/etc/e-smith/tests/10e-smith-lib/db_dummy.conf
Normal file
7
root/etc/e-smith/tests/10e-smith-lib/db_dummy.conf
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
# Dummy configuration file for testing esmith::db
|
||||||
|
Foo=Bar
|
||||||
|
Night=Day
|
||||||
|
Squid=cephalopod|arms|10|species|Loligo
|
||||||
|
Pipe=art|pipe|this is not a \| got that?
|
||||||
|
Haiku=poem|words|Damian Conway\nGod damn! Damian Conway\nDamian Conway
|
||||||
|
Octopus=cephalopod|arms|8|species|
|
141
root/etc/e-smith/tests/10e-smith-lib/default_migrate_force.t
Normal file
141
root/etc/e-smith/tests/10e-smith-lib/default_migrate_force.t
Normal file
@ -0,0 +1,141 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
# vim: ft=perl:
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use esmith::util;
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
use Test::More 'no_plan';
|
||||||
|
use File::Copy qw(copy);
|
||||||
|
use POSIX qw(tmpnam);
|
||||||
|
|
||||||
|
# What we want to do is run initialize-default-databases on a scratch
|
||||||
|
# db and check the results.
|
||||||
|
my $dbhome = '/tmp/dbhome';
|
||||||
|
my $dbroot = '/etc/e-smith/db';
|
||||||
|
system('rm', '-rf', $dbhome);
|
||||||
|
system('mkdir', '-p', $dbhome) == 0
|
||||||
|
or die "Can't create $dbhome: $!\n";
|
||||||
|
|
||||||
|
ok( esmith::util::initialize_default_databases(dbhome => $dbhome),
|
||||||
|
"initialize_default_databases created successfully" );
|
||||||
|
|
||||||
|
# Confirm the default settings.
|
||||||
|
walk_dbtree($dbroot, 'defaults');
|
||||||
|
# Test that forced options were forced.
|
||||||
|
walk_dbtree($dbroot, 'force');
|
||||||
|
|
||||||
|
# We should now have default configuration files. We can go through each
|
||||||
|
# setting specified by the defaults and confirm that it is correct.
|
||||||
|
|
||||||
|
# To test migration, we should take a typical 5.6 set of databases and migrate
|
||||||
|
# those. The existing .conf databases in the 10e-smith-lib directory are
|
||||||
|
# styled after 5.6, and suitable for this.
|
||||||
|
foreach my $dummyconf (qw(accounts.conf domains.conf networks.conf
|
||||||
|
configuration.conf hosts.conf))
|
||||||
|
{
|
||||||
|
# Copy them over the ones in our test directory, and then migrate them.
|
||||||
|
my $dest;
|
||||||
|
($dest = $dummyconf) =~ s/\.conf$//;
|
||||||
|
$dest = "$dbhome/$dest";
|
||||||
|
copy($dummyconf, $dest) or die "Can't copy $dummyconf to $dest: $!\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
# Migrate the directory.
|
||||||
|
ok( esmith::util::initialize_default_databases(dbhome => $dbhome),
|
||||||
|
"initialize_default_databases migrated successfully" );
|
||||||
|
|
||||||
|
#run_migrate_tests($dbhome, $dbroot);
|
||||||
|
system('rm', '-rf', $dbhome);
|
||||||
|
|
||||||
|
exit 0;
|
||||||
|
|
||||||
|
sub walk_dbtree
|
||||||
|
{
|
||||||
|
my $dbroot = shift;
|
||||||
|
my $mode = shift;
|
||||||
|
die unless $mode =~ /^(defaults|force)$/;
|
||||||
|
|
||||||
|
opendir(DBROOT, $dbroot) or die "Can't open $dbroot: $!\n";
|
||||||
|
my @dbfiles = grep { -d "$dbroot/$_" }
|
||||||
|
grep { !/^\./ } readdir DBROOT;
|
||||||
|
closedir(DBROOT);
|
||||||
|
foreach my $dbfile (@dbfiles)
|
||||||
|
{
|
||||||
|
# Handle the defaults in this case.
|
||||||
|
my $defaultdir = "$dbroot/$dbfile/defaults";
|
||||||
|
next if not -e $defaultdir;
|
||||||
|
opendir(DEFAULTDIR, $defaultdir) or die "Can't open $defaultdir: $!\n";
|
||||||
|
my @keys = grep { -d "$defaultdir/$_" }
|
||||||
|
grep { !/^\./ } readdir DEFAULTDIR;
|
||||||
|
closedir(DEFAULTDIR);
|
||||||
|
# There should now be a db file output at the dbhome.
|
||||||
|
ok( -e "$dbhome/$dbfile", "$dbfile db exists" );
|
||||||
|
|
||||||
|
my $db = esmith::ConfigDB->open_ro("$dbhome/$dbfile");
|
||||||
|
ok( defined $db, "$dbhome/$dbfile loads properly" );
|
||||||
|
# Loop on all the keys.
|
||||||
|
foreach my $keydir (@keys)
|
||||||
|
{
|
||||||
|
my $key_fullpath = "$defaultdir/$keydir";
|
||||||
|
opendir(KEYDIR, $key_fullpath)
|
||||||
|
or die "Can't open $key_fullpath: $!\n";
|
||||||
|
my @propfiles = grep { -f "$key_fullpath/$_" }
|
||||||
|
grep { !/^\./ } readdir KEYDIR;
|
||||||
|
closedir(KEYDIR);
|
||||||
|
|
||||||
|
foreach my $propfile (@propfiles)
|
||||||
|
{
|
||||||
|
my $propfile_fullpath = "$key_fullpath/$propfile";
|
||||||
|
# Open each and check what the default should be.
|
||||||
|
open(PROPFILE, "<$propfile_fullpath")
|
||||||
|
or die "Can't open $propfile_fullpath: $!\n";
|
||||||
|
chomp( my $propval = <PROPFILE> );
|
||||||
|
close(PROPFILE);
|
||||||
|
if ($keydir eq 'ActiveAccounts')
|
||||||
|
{
|
||||||
|
print "get_prop on $keydir, $propfile returns ";
|
||||||
|
print $db->get_prop($keydir, $propfile) . "\n";
|
||||||
|
print "propval is $propval\n";
|
||||||
|
}
|
||||||
|
ok( $db->get_prop($keydir, $propfile) eq $propval,
|
||||||
|
"property $propfile of record $keydir has correct $mode value of $propval" );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
# FIXME: This works, but the numbers of the tests are thrown off. We probably
|
||||||
|
# want to eval the test block of code instead.
|
||||||
|
sub run_migrate_tests
|
||||||
|
{
|
||||||
|
my $dbhome = shift;
|
||||||
|
my $dbroot = shift;
|
||||||
|
|
||||||
|
opendir(DBROOT, $dbroot) or die "Can't open $dbroot: $!\n";
|
||||||
|
my @dbfiles = grep { -d "$dbroot/$_" }
|
||||||
|
grep { !/^\./ } readdir DBROOT;
|
||||||
|
closedir(DBROOT);
|
||||||
|
foreach my $dbfile (@dbfiles)
|
||||||
|
{
|
||||||
|
# Handle the defaults in this case.
|
||||||
|
my $migratedir = "$dbroot/$dbfile/migrate";
|
||||||
|
next if not -e $migratedir;
|
||||||
|
|
||||||
|
opendir(MIGRATEDIR, $migratedir) or die "Can't open $migratedir: $!\n";
|
||||||
|
my @migrate_fragments = grep { !/^\./ } readdir MIGRATEDIR;
|
||||||
|
closedir(MIGRATEDIR);
|
||||||
|
|
||||||
|
foreach my $migrate_fragment (sort @migrate_fragments)
|
||||||
|
{
|
||||||
|
my $tempname = tmpnam() or die "Can't obtain tempfile: $!\n";
|
||||||
|
my $pod2test = '/usr/bin/pod2test';
|
||||||
|
system($pod2test, "$migratedir/$migrate_fragment", $tempname);
|
||||||
|
if (! -e $tempname)
|
||||||
|
{
|
||||||
|
warn "The fragment $migrate_fragment apparently has no embedded tests\n";
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
system('/usr/bin/perl', $tempname);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
3
root/etc/e-smith/tests/10e-smith-lib/domains.conf
Normal file
3
root/etc/e-smith/tests/10e-smith-lib/domains.conf
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
test=domain|foo|bar
|
||||||
|
foo=domain|baz|quux
|
||||||
|
wombat=notadomain
|
11
root/etc/e-smith/tests/10e-smith-lib/dummy.conf
Normal file
11
root/etc/e-smith/tests/10e-smith-lib/dummy.conf
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
foo=bar
|
||||||
|
this key=has whitespace
|
||||||
|
that key =has trailing whitespace
|
||||||
|
another key=has leading whitespace
|
||||||
|
this value= has leading whitespace
|
||||||
|
that value=has trailing whitespace
|
||||||
|
|
||||||
|
# this is a comment. ignore it.
|
||||||
|
# this is a comment = too
|
||||||
|
|
||||||
|
tricky value=with=equals.
|
13
root/etc/e-smith/tests/10e-smith-lib/hosts.conf
Normal file
13
root/etc/e-smith/tests/10e-smith-lib/hosts.conf
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# DO NOT MODIFY THIS FILE.
|
||||||
|
# This file is automatically maintained by the Mitel Networks SME Server
|
||||||
|
# configuration software. Manually editing this file may put your
|
||||||
|
# system in an unknown state.
|
||||||
|
#
|
||||||
|
# updated: Fri Mar 11 18:21:42 2002
|
||||||
|
ftp.mydomain.xxx=host|ExternalIP||HostType|Self|InternalIP||MACAddress||Visibility|Local
|
||||||
|
mail.mydomain.xxx=host|ExternalIP||HostType|Self|InternalIP||MACAddress||Visibility|Local
|
||||||
|
myserver.mydomain.xxx=host|ExternalIP||HostType|Local|InternalIP||MACAddress||Visibility|Local
|
||||||
|
otherhost.mydomain.xxx=host|ExternalIP||HostType|Local|InternalIP|192.168.1.3|MACAddress||Visibility|Local
|
||||||
|
www.mydomain.xxx=host|ExternalIP||HostType|Self|InternalIP||MACAddress||Visibility|Local
|
||||||
|
www.otherdomain.xxx=host|ExternalIP||HostType|Self|InternalIP||MACAddress||Visibility|Local
|
||||||
|
foo.otherdomain.xxx=host|ExternalIP||HostType|Self|InternalIP||MACAddress||Visibility|Local
|
@ -0,0 +1 @@
|
|||||||
|
I am an English license.
|
@ -0,0 +1 @@
|
|||||||
|
Je suis une license francais. Or something like that.
|
1
root/etc/e-smith/tests/10e-smith-lib/networks.conf
Normal file
1
root/etc/e-smith/tests/10e-smith-lib/networks.conf
Normal file
@ -0,0 +1 @@
|
|||||||
|
10.0.0.0=network|Mask|255.255.255.0|Router|default
|
@ -0,0 +1 @@
|
|||||||
|
sysconfig=configuration|KeyboardType|pc|Keytable|us|Language|en_US|ReleaseVersion|6.0alpha2
|
@ -0,0 +1 @@
|
|||||||
|
sysconfig=configuration|KeyboardType|pc|Keytable|us|Language|fr_CA|ReleaseVersion|6.0alpha2
|
111
root/etc/e-smith/tests/10e-smith-lib/templates.t
Normal file
111
root/etc/e-smith/tests/10e-smith-lib/templates.t
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
|
||||||
|
use esmith::TestUtils qw(scratch_copy);
|
||||||
|
use File::Path;
|
||||||
|
|
||||||
|
use Test::More 'no_plan';
|
||||||
|
|
||||||
|
use_ok('esmith::templates', qw(:DEFAULT removeBlankLines));
|
||||||
|
|
||||||
|
use esmith::config;
|
||||||
|
my %config;
|
||||||
|
tie %config, 'esmith::config', '10e-smith-lib/scratch.conf';
|
||||||
|
$ENV{ESMITH_CONFIG_DB} = '10e-smith-lib/scratch.conf';
|
||||||
|
END { unlink '10e-smith-lib/scratch.conf' }
|
||||||
|
|
||||||
|
my $Scratch_Temp_Dir = 'template_scratch_dir';
|
||||||
|
my $Scratch_Temp = "$Scratch_Temp_Dir/dummy";
|
||||||
|
mkpath "$Scratch_Temp_Dir/10e-smith-lib";
|
||||||
|
END { rmtree $Scratch_Temp_Dir }
|
||||||
|
|
||||||
|
# processTemplate() is going to be Loud and Helpful about skipping
|
||||||
|
# things like CVS directories.
|
||||||
|
$SIG{__WARN__} = sub { return if $_[0] =~ /^Skipping directory/ };
|
||||||
|
|
||||||
|
$config{Koala_Say} = "This is not the bear you're looking for.";
|
||||||
|
processTemplate({ CONFREF => \%config,
|
||||||
|
OUTPUT_PREFIX => $Scratch_Temp_Dir,
|
||||||
|
OUTPUT_FILENAME => 'dummy',
|
||||||
|
TEMPLATE_PATH => 'templates',
|
||||||
|
TEMPLATE_EXPAND_QUEUE => [
|
||||||
|
'10e-smith-lib'
|
||||||
|
],
|
||||||
|
FILTER => \&removeBlankLines,
|
||||||
|
UID => $<,
|
||||||
|
GID => (split / /, $()[0]
|
||||||
|
});
|
||||||
|
|
||||||
|
ok( -f $Scratch_Temp, 'file generated' );
|
||||||
|
ok( -s $Scratch_Temp, ' its not empty' );
|
||||||
|
|
||||||
|
open(SCRATCH, $Scratch_Temp) || die $!;
|
||||||
|
my $out;
|
||||||
|
{ local $/; $out = <SCRATCH>; }
|
||||||
|
close SCRATCH;
|
||||||
|
|
||||||
|
my $koala_output = <<'THIS';
|
||||||
|
# This is the beginning of the beginning
|
||||||
|
# confref ok
|
||||||
|
----------------------------------------
|
||||||
|
< This is not the bear you're looking for. >
|
||||||
|
----------------------------------------
|
||||||
|
\
|
||||||
|
\ .
|
||||||
|
___ //
|
||||||
|
{~._.~}//
|
||||||
|
( Y )K/
|
||||||
|
()~*~()
|
||||||
|
(_)-(_)
|
||||||
|
Luke
|
||||||
|
Skywalker
|
||||||
|
koala
|
||||||
|
# This is the end, My only friend, the end of our elaborate templates, the end
|
||||||
|
THIS
|
||||||
|
|
||||||
|
is( $out, $koala_output, 'file generated properly' );
|
||||||
|
|
||||||
|
$out = processTemplate({
|
||||||
|
CONFREF => \%config,
|
||||||
|
TEMPLATE_PATH => 'templates',
|
||||||
|
TEMPLATE_EXPAND_QUEUE => [
|
||||||
|
'10e-smith-lib'
|
||||||
|
],
|
||||||
|
FILTER => sub { $_[0] =~ /^\s*$/ ? '' : $_[0] },
|
||||||
|
UID => $<,
|
||||||
|
GID => (split / /, $()[0],
|
||||||
|
OUTPUT_TYPE => 'string'
|
||||||
|
});
|
||||||
|
|
||||||
|
is( $out, $koala_output, 'string generated properly' );
|
||||||
|
|
||||||
|
my $h_scratch = scratch_copy('10e-smith-lib/hosts.conf');
|
||||||
|
my $a_scratch = scratch_copy('10e-smith-lib/accounts.conf');
|
||||||
|
use esmith::AccountsDB;
|
||||||
|
use esmith::HostsDB;
|
||||||
|
my $acct = esmith::AccountsDB->open($a_scratch);
|
||||||
|
my $host = esmith::HostsDB->open($h_scratch);
|
||||||
|
$out = processTemplate({
|
||||||
|
MORE_DATA => { Author => 'Douglas Adams' },
|
||||||
|
TEMPLATE_PATH => 'templates_DB',
|
||||||
|
TEMPLATE_EXPAND_QUEUE => [
|
||||||
|
'10e-smith-lib'
|
||||||
|
],
|
||||||
|
OUTPUT_TYPE => 'string'
|
||||||
|
});
|
||||||
|
is( $out, <<'THIS', 'DB & MORE_DATA' );
|
||||||
|
Chapter 1
|
||||||
|
|
||||||
|
The story so far:
|
||||||
|
|
||||||
|
In the beginning the Universe was created. This has made a lot
|
||||||
|
of people very angry and been widely regarded as a bad move.
|
||||||
|
-- Douglas Adams
|
||||||
|
|
||||||
|
$DB ok
|
||||||
|
default vars ok
|
||||||
|
|
||||||
|
confref not defined
|
||||||
|
|
||||||
|
The end of labor is to gain leisure.
|
||||||
|
THIS
|
||||||
|
|
20
root/etc/e-smith/tests/10e-smith-lib/templates/10moof
Normal file
20
root/etc/e-smith/tests/10e-smith-lib/templates/10moof
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
my $ksez = $Koala_Say;
|
||||||
|
my $line = '-' x length $ksez;
|
||||||
|
|
||||||
|
$OUT = <<KOALA_SEZ;
|
||||||
|
$line
|
||||||
|
< $ksez >
|
||||||
|
$line
|
||||||
|
KOALA_SEZ
|
||||||
|
}
|
||||||
|
\
|
||||||
|
\ .
|
||||||
|
___ //
|
||||||
|
\{~._.~\}//
|
||||||
|
( Y )K/
|
||||||
|
()~*~()
|
||||||
|
(_)-(_)
|
||||||
|
Luke
|
||||||
|
Skywalker
|
||||||
|
koala
|
@ -0,0 +1,3 @@
|
|||||||
|
# This is the beginning of the beginning
|
||||||
|
# { keys %$confref == 1 && exists $confref->{Koala_Say}
|
||||||
|
? "confref ok" : "confref not ok" }
|
@ -0,0 +1 @@
|
|||||||
|
# This is the end, My only friend, the end of our elaborate templates, the end
|
20
root/etc/e-smith/tests/10e-smith-lib/templates2/10moof
Normal file
20
root/etc/e-smith/tests/10e-smith-lib/templates2/10moof
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
my $ksez = $Koala_Say;
|
||||||
|
my $line = '-' x length $ksez;
|
||||||
|
|
||||||
|
$OUT = <<KOALA_SEZ;
|
||||||
|
$line
|
||||||
|
< $ksez >
|
||||||
|
$line
|
||||||
|
KOALA_SEZ
|
||||||
|
}
|
||||||
|
\
|
||||||
|
\ .
|
||||||
|
___ //
|
||||||
|
\{~._.~\}//
|
||||||
|
( Y )K/
|
||||||
|
()~*~()
|
||||||
|
(_)-(_)
|
||||||
|
Luke
|
||||||
|
Skywalker
|
||||||
|
koala
|
@ -0,0 +1 @@
|
|||||||
|
# This is the end, My only friend, the end of our elaborate templates, the end
|
9
root/etc/e-smith/tests/10e-smith-lib/templates_DB/10DB
Normal file
9
root/etc/e-smith/tests/10e-smith-lib/templates_DB/10DB
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
{ '$DB ok' if defined $DB && $DB->isa('esmith::ConfigDB') }
|
||||||
|
{ my $ok = 1;
|
||||||
|
foreach my $rec ($DB->get_all) {
|
||||||
|
my $key = $rec->key;
|
||||||
|
my $type = $rec->props <= 1 ? "SCALAR" : "HASH";
|
||||||
|
$ok = 0 unless *{$key}{$type};
|
||||||
|
}
|
||||||
|
'default vars ok' if $ok;
|
||||||
|
}
|
@ -0,0 +1,7 @@
|
|||||||
|
|
||||||
|
{ if( *{confref}{SCALAR} ) {
|
||||||
|
"confref not defined";
|
||||||
|
} else {
|
||||||
|
"confref defined"
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,10 @@
|
|||||||
|
Chapter 1
|
||||||
|
|
||||||
|
The story so far:
|
||||||
|
|
||||||
|
In the beginning the Universe was created. This has made a lot
|
||||||
|
of people very angry and been widely regarded as a bad move.
|
||||||
|
-- { # Testing MORE_DATA
|
||||||
|
$Author
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,2 @@
|
|||||||
|
|
||||||
|
The end of labor is to gain leisure.
|
112
root/etc/e-smith/tests/10e-smith-lib/uidgid.conf
Normal file
112
root/etc/e-smith/tests/10e-smith-lib/uidgid.conf
Normal file
@ -0,0 +1,112 @@
|
|||||||
|
# This file stores uid/gid information from the CreatingSystemUsers topic of
|
||||||
|
# the Engineering Wiki. It is read by the uidgid.t test script. Please keep it
|
||||||
|
# up to date.
|
||||||
|
[passwd]
|
||||||
|
0 root 0
|
||||||
|
1 bin 1
|
||||||
|
2 daemon 2
|
||||||
|
3 adm 4
|
||||||
|
4 lp 7
|
||||||
|
8 mail 12
|
||||||
|
9 news 13
|
||||||
|
10 uucp 14
|
||||||
|
11 operator 0
|
||||||
|
12 games 100
|
||||||
|
13 gopher 30
|
||||||
|
14 ftp 50
|
||||||
|
23 squid 23
|
||||||
|
27 mysql 27
|
||||||
|
32 rpc 32
|
||||||
|
37 rpm 37
|
||||||
|
38 ntp 38
|
||||||
|
48 apache 48
|
||||||
|
53 dns 53
|
||||||
|
55 ldap 55
|
||||||
|
69 vcsa 69
|
||||||
|
74 sshd 74
|
||||||
|
77 pcap 77
|
||||||
|
99 nobody 99
|
||||||
|
100 www 101
|
||||||
|
101 admin 101
|
||||||
|
102 public 103
|
||||||
|
400 alias 400
|
||||||
|
401 qmaild 400
|
||||||
|
402 qmaill 400
|
||||||
|
403 qmailp 400
|
||||||
|
404 qmailq 401
|
||||||
|
405 qmailr 401
|
||||||
|
406 qmails 401
|
||||||
|
407 qmailscan 407
|
||||||
|
408 jabber 408
|
||||||
|
410 dnscache 410
|
||||||
|
411 dnslog 411
|
||||||
|
420 nutups 420
|
||||||
|
421 trend 421
|
||||||
|
422 fax 422
|
||||||
|
451 stunnel 451
|
||||||
|
452 memtestlog 452
|
||||||
|
1001 imaplog 1001
|
||||||
|
1002 smelog 1002
|
||||||
|
1003 cvmlog 1003
|
||||||
|
1004 mn_dvr 1004
|
||||||
|
1005 spamd 1005
|
||||||
|
2999 smelastsys 2999
|
||||||
|
|
||||||
|
[group]
|
||||||
|
0 root
|
||||||
|
1 bin
|
||||||
|
2 daemon
|
||||||
|
3 sys
|
||||||
|
4 adm
|
||||||
|
5 tty
|
||||||
|
6 disk
|
||||||
|
7 lp
|
||||||
|
8 mem
|
||||||
|
9 kmem
|
||||||
|
10 wheel
|
||||||
|
12 mail
|
||||||
|
13 news
|
||||||
|
14 uucp
|
||||||
|
15 man
|
||||||
|
19 floppy
|
||||||
|
20 games
|
||||||
|
21 slocate
|
||||||
|
22 utmp
|
||||||
|
23 squid
|
||||||
|
27 mysql
|
||||||
|
30 gopher
|
||||||
|
32 rpc
|
||||||
|
37 rpm
|
||||||
|
38 ntp
|
||||||
|
40 dip
|
||||||
|
48 apache
|
||||||
|
50 ftp
|
||||||
|
53 dns
|
||||||
|
54 lock
|
||||||
|
55 ldap
|
||||||
|
69 vcsa
|
||||||
|
74 sshd
|
||||||
|
77 pcap
|
||||||
|
99 nobody
|
||||||
|
100 users
|
||||||
|
101 admin
|
||||||
|
102 www
|
||||||
|
103 public
|
||||||
|
400 nofiles
|
||||||
|
401 qmail
|
||||||
|
407 qmailscan
|
||||||
|
408 jabber
|
||||||
|
410 dnscache
|
||||||
|
411 dnslog
|
||||||
|
420 nutups
|
||||||
|
421 trend
|
||||||
|
422 fax
|
||||||
|
451 stunnel
|
||||||
|
452 memtestlog
|
||||||
|
500 shared
|
||||||
|
1001 imaplog
|
||||||
|
1002 smelog
|
||||||
|
1003 cvmlog
|
||||||
|
1004 mn_dvr
|
||||||
|
1005 spamd
|
||||||
|
2999 smelastsys
|
78
root/etc/e-smith/tests/10e-smith-lib/uidgid.t
Normal file
78
root/etc/e-smith/tests/10e-smith-lib/uidgid.t
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
# vim: ft=perl:
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use Test::More 'no_plan';
|
||||||
|
use Unix::PasswdFile;
|
||||||
|
use Unix::GroupFile;
|
||||||
|
|
||||||
|
use constant TRUE => 1;
|
||||||
|
use constant FALSE => 0;
|
||||||
|
|
||||||
|
my $conffile = 'uidgid.conf';
|
||||||
|
|
||||||
|
exit 1 if not -e $conffile;
|
||||||
|
|
||||||
|
my %passwdlist = ();
|
||||||
|
my %grouplist = ();
|
||||||
|
my $passwd = FALSE;
|
||||||
|
my $group = FALSE;
|
||||||
|
|
||||||
|
open(CONF, "<$conffile") or die "Can't open $conffile: $!\n";
|
||||||
|
|
||||||
|
while(<CONF>)
|
||||||
|
{
|
||||||
|
next if /^(#|\s)/;
|
||||||
|
if (/\[passwd]/)
|
||||||
|
{
|
||||||
|
$passwd = TRUE;
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
elsif (/\[group]/)
|
||||||
|
{
|
||||||
|
$group = TRUE;
|
||||||
|
$passwd = FALSE;
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
next if not $passwd and not $group;
|
||||||
|
|
||||||
|
if ($passwd)
|
||||||
|
{
|
||||||
|
my ($uid, $name, $gid) = split;
|
||||||
|
$passwdlist{$name}{uid} = $uid;
|
||||||
|
$passwdlist{$name}{gid} = $gid;
|
||||||
|
}
|
||||||
|
elsif ($group)
|
||||||
|
{
|
||||||
|
my ($gid, $name) = split;
|
||||||
|
$grouplist{$name} = $gid;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
close(CONF);
|
||||||
|
|
||||||
|
# We can now confirm the uid and gid of every user, and the gid of every
|
||||||
|
# group, on the system.
|
||||||
|
# I don't know of a good way to query every user on the system, so for now
|
||||||
|
# lets just read them from the passwd file.
|
||||||
|
my $pw = Unix::PasswdFile->new('/etc/passwd', mode => 'r')
|
||||||
|
or die "Can't open /etc/passwd: $!\n";
|
||||||
|
|
||||||
|
foreach my $user ($pw->users)
|
||||||
|
{
|
||||||
|
my ($name,$passwd,$uid,$gid,undef) = getpwnam($user);
|
||||||
|
ok( exists $passwdlist{$name}, "user $name is on our list" );
|
||||||
|
ok( $uid == $passwdlist{$name}{uid}, "user $name has uid of $uid" );
|
||||||
|
ok( $gid == $passwdlist{$name}{gid}, "user $name has gid of $gid" );
|
||||||
|
}
|
||||||
|
|
||||||
|
my $gr = Unix::GroupFile->new('/etc/group', mode => 'r')
|
||||||
|
or die "Can't open /etc/group: $!\n";
|
||||||
|
|
||||||
|
foreach my $group ($gr->groups)
|
||||||
|
{
|
||||||
|
my ($name,$passwd,$gid,$members) = getgrnam($group);
|
||||||
|
ok( exists $grouplist{$name}, "group $name is on our list" );
|
||||||
|
ok( $gid == $grouplist{$name}, "group $group has gid of $gid" );
|
||||||
|
}
|
||||||
|
|
||||||
|
exit 0;
|
10
root/sbin/e-smith/config
Executable file
10
root/sbin/e-smith/config
Executable file
@ -0,0 +1,10 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
exec "/sbin/e-smith/db", "configuration", @ARGV;
|
||||||
|
die "Could not exec /sbin/e-smith/db";
|
84
root/sbin/e-smith/create-system-user
Normal file
84
root/sbin/e-smith/create-system-user
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
#! /usr/bin/perl -w
|
||||||
|
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
|
||||||
|
sub usage
|
||||||
|
{
|
||||||
|
my $msg = shift;
|
||||||
|
warn("$msg\n") if $msg;
|
||||||
|
die("Usage: $0: user userid descr home_dir shell\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
my $user = shift || usage("Must give username param");
|
||||||
|
my $uid = shift || usage("Must give userid param");
|
||||||
|
my $user_desc = shift || usage("Must give user desc param");
|
||||||
|
my $home = shift || usage("Must give home param");
|
||||||
|
my $shell = shift || usage("Must give shell param");
|
||||||
|
|
||||||
|
use User::pwent;
|
||||||
|
use User::grent;
|
||||||
|
|
||||||
|
if (my $pw = getpwnam($user))
|
||||||
|
{
|
||||||
|
my $euid = $pw->uid;
|
||||||
|
exit 0 if $euid == $uid; # Do not create user if it already exists with correct uid
|
||||||
|
warn ("Users $user exists but has uid of $euid - should be $uid\n");
|
||||||
|
exit 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (my $pw = getpwuid($uid))
|
||||||
|
{
|
||||||
|
my $name = $pw->name;
|
||||||
|
warn "User id of $uid is already taken by user $name\n";
|
||||||
|
warn "Falling back to a system chosen uid\n";
|
||||||
|
# We can now go ahead and create the user and group leaving the system to choose uid/gid
|
||||||
|
die ("Failed to create user $user\n") if
|
||||||
|
system("/usr/sbin/useradd",
|
||||||
|
"-r",
|
||||||
|
"-d", $home,
|
||||||
|
"-M",
|
||||||
|
"-s", $shell,
|
||||||
|
"-c", $user_desc,
|
||||||
|
$user);
|
||||||
|
exit (0);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (my $pw = getgrgid($uid))
|
||||||
|
{
|
||||||
|
my $name = $pw->name;
|
||||||
|
warn "Group id of $uid is already taken by user $name\n";
|
||||||
|
# We can now go ahead and create the user but the gid will be chosen by the system
|
||||||
|
die ("Failed to create user $user\n") if
|
||||||
|
system("/usr/sbin/useradd",
|
||||||
|
"-u", $uid,
|
||||||
|
"-d", $home,
|
||||||
|
"-M",
|
||||||
|
"-s", $shell,
|
||||||
|
"-c", $user_desc,
|
||||||
|
$user);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
# We can now go ahead and create the user and group
|
||||||
|
die ("Failed to create group $uid\n") if
|
||||||
|
system("/usr/sbin/groupadd",
|
||||||
|
"-g", $uid,
|
||||||
|
$user);
|
||||||
|
|
||||||
|
die ("Failed to create user $user\n") if
|
||||||
|
system("/usr/sbin/useradd",
|
||||||
|
"-u", $uid,
|
||||||
|
"-g", $uid,
|
||||||
|
"-d", $home,
|
||||||
|
"-M",
|
||||||
|
"-s", $shell,
|
||||||
|
"-c", $user_desc,
|
||||||
|
$user);
|
||||||
|
}
|
||||||
|
exit 0;
|
344
root/sbin/e-smith/db
Normal file
344
root/sbin/e-smith/db
Normal file
@ -0,0 +1,344 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use esmith::config;
|
||||||
|
use esmith::db;
|
||||||
|
|
||||||
|
#--------------------------------------------------------------------------
|
||||||
|
# Set up the command list and usage strings
|
||||||
|
|
||||||
|
my %commands = (
|
||||||
|
'keys'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_keys,
|
||||||
|
'usage' => "$0 dbfile keys",
|
||||||
|
},
|
||||||
|
|
||||||
|
'print'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_print,
|
||||||
|
'usage' => "$0 dbfile print [key]",
|
||||||
|
},
|
||||||
|
|
||||||
|
'show'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_show,
|
||||||
|
'usage' => "$0 dbfile show [key]",
|
||||||
|
},
|
||||||
|
|
||||||
|
'get'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_get,
|
||||||
|
'usage' => "$0 dbfile get key",
|
||||||
|
},
|
||||||
|
|
||||||
|
'set'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_set,
|
||||||
|
'usage' => "$0 dbfile set key type "
|
||||||
|
. "[prop1 val1] [prop2 val2] ...",
|
||||||
|
},
|
||||||
|
|
||||||
|
'setdefault'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_set_default,
|
||||||
|
'usage' => "$0 dbfile setdefault key type "
|
||||||
|
. "[prop1 val1] [prop2 val2] ...",
|
||||||
|
},
|
||||||
|
|
||||||
|
'delete'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_delete,
|
||||||
|
'usage' => "$0 dbfile delete key",
|
||||||
|
},
|
||||||
|
|
||||||
|
'printtype'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_printtype,
|
||||||
|
'usage' => "$0 dbfile printtype [key]",
|
||||||
|
},
|
||||||
|
|
||||||
|
'gettype'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_gettype,
|
||||||
|
'usage' => "$0 dbfile gettype key",
|
||||||
|
},
|
||||||
|
|
||||||
|
'settype'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_settype,
|
||||||
|
'usage' => "$0 dbfile settype key type",
|
||||||
|
},
|
||||||
|
|
||||||
|
'printprop'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_printprop,
|
||||||
|
'usage' => "$0 dbfile printprop key [prop1] "
|
||||||
|
. "[prop2] [prop3] ...",
|
||||||
|
},
|
||||||
|
|
||||||
|
'getprop'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_getprop,
|
||||||
|
'usage' => "$0 dbfile getprop key prop",
|
||||||
|
},
|
||||||
|
|
||||||
|
'setprop'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_setprop,
|
||||||
|
'usage' => "$0 dbfile setprop key "
|
||||||
|
. "prop1 val1 [prop2 val2] "
|
||||||
|
. "[prop3 val3] ...",
|
||||||
|
},
|
||||||
|
|
||||||
|
'delprop'
|
||||||
|
=> {
|
||||||
|
'function' => \&DB_delprop,
|
||||||
|
'usage' => "$0 dbfile delprop key prop1 "
|
||||||
|
. "[prop2] [prop3] ...",
|
||||||
|
},
|
||||||
|
|
||||||
|
);
|
||||||
|
|
||||||
|
#--------------------------------------------------------------------------
|
||||||
|
# Set up general usage message.
|
||||||
|
|
||||||
|
my $usage = "usage:
|
||||||
|
$commands{'keys'}{'usage'}
|
||||||
|
$commands{'print'}{'usage'}
|
||||||
|
$commands{'show'}{'usage'}
|
||||||
|
$commands{'get'}{'usage'}
|
||||||
|
$commands{'set'}{'usage'}
|
||||||
|
$commands{'setdefault'}{'usage'}
|
||||||
|
$commands{'delete'}{'usage'}
|
||||||
|
$commands{'printtype'}{'usage'}
|
||||||
|
$commands{'gettype'}{'usage'}
|
||||||
|
$commands{'settype'}{'usage'}
|
||||||
|
$commands{'printprop'}{'usage'}
|
||||||
|
$commands{'getprop'}{'usage'}
|
||||||
|
$commands{'setprop'}{'usage'}
|
||||||
|
$commands{'delprop'}{'usage'}
|
||||||
|
";
|
||||||
|
|
||||||
|
#--------------------------------------------------------------------------
|
||||||
|
# Prepend $ENV{'DBPATH'} to dbfile if defined otherwise let the library
|
||||||
|
# deal with it.
|
||||||
|
|
||||||
|
my $dbfile = shift;
|
||||||
|
die $usage unless $dbfile;
|
||||||
|
|
||||||
|
my $dbpath = $ENV{'DBPATH'};
|
||||||
|
$dbfile = "$dbpath/$dbfile" if defined $dbpath;
|
||||||
|
|
||||||
|
#--------------------------------------------------------------------------
|
||||||
|
# Tie the databasefile to a hash
|
||||||
|
|
||||||
|
my $db = esmith::db->open($dbfile);
|
||||||
|
|
||||||
|
#--------------------------------------------------------------------------
|
||||||
|
# Run the appropriate command
|
||||||
|
|
||||||
|
my $command = shift;
|
||||||
|
die $usage unless $command;
|
||||||
|
die $usage unless exists $commands{$command};
|
||||||
|
$commands{$command}{'function'}->(@ARGV);
|
||||||
|
|
||||||
|
exit 0;
|
||||||
|
|
||||||
|
#--------------------------------------------------------------------------
|
||||||
|
|
||||||
|
sub DB_print
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
|
||||||
|
if (defined $key)
|
||||||
|
{
|
||||||
|
$db->print($key) ? exit 0 : exit 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$db->print() ? exit 0 : exit 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_keys
|
||||||
|
{
|
||||||
|
my @keys = $db->get();
|
||||||
|
|
||||||
|
exit 1 unless (scalar @keys);
|
||||||
|
print join("\n", @keys), "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_show
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
|
||||||
|
if (defined $key)
|
||||||
|
{
|
||||||
|
$db->show($key) ? exit 0 : exit 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$db->show() ? exit 0 : exit 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_get
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
die "$commands{'get'}{'usage'}\n" unless $key;
|
||||||
|
|
||||||
|
my $value = $db->get($key);
|
||||||
|
exit 1 unless defined $value;
|
||||||
|
print "$value\n" if defined $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_set
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
die "$commands{'set'}{'usage'}\n" unless $key;
|
||||||
|
die "$commands{'set'}{'usage'}\n" unless scalar @_;
|
||||||
|
|
||||||
|
my $type = shift;
|
||||||
|
die "$commands{'set'}{'usage'}\n" unless defined $type;
|
||||||
|
die "$commands{'set'}{'usage'}\n" if scalar @_ % 2;
|
||||||
|
|
||||||
|
$db->set($key, $type) or exit 1;
|
||||||
|
|
||||||
|
&DB_setprop($key, @_) if scalar @_;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_set_default
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
die "$commands{'setdefault'}{'usage'}\n" unless $key;
|
||||||
|
die "$commands{'setdefault'}{'usage'}\n" unless scalar @_;
|
||||||
|
|
||||||
|
my $type = shift;
|
||||||
|
die "$commands{'setdefault'}{'usage'}\n" unless $type;
|
||||||
|
die "$commands{'setdefault'}{'usage'}\n" if scalar @_ % 2;
|
||||||
|
|
||||||
|
# Only set values if the key does not exist
|
||||||
|
|
||||||
|
exit 0 if defined $db->get($key);
|
||||||
|
|
||||||
|
&DB_set($key, $type, @_);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_delete
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
die "$commands{'delete'}{'usage'}\n" unless $key;
|
||||||
|
|
||||||
|
$db->delete($key) ? exit 0 : exit 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_printtype
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
|
||||||
|
if (defined $key)
|
||||||
|
{
|
||||||
|
$db->print_type($key) ? exit 0 : exit 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$db->print_type() ? exit 0 : exit 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_gettype
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
die "$commands{'get'}{'usage'}\n" unless $key;
|
||||||
|
|
||||||
|
my $value = $db->get_type($key);
|
||||||
|
exit 1 unless defined $value;
|
||||||
|
print "$value\n" if defined $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_settype
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
die "$commands{'settype'}{'usage'}\n" unless $key;
|
||||||
|
my $type = shift;
|
||||||
|
die "$commands{'settype'}{'usage'}\n" unless $type;
|
||||||
|
|
||||||
|
$db->set_type($key, $type) ? exit 0 : exit 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_printprop
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
die "$commands{'printprop'}{'usage'}\n" unless $key;
|
||||||
|
|
||||||
|
my @props = @_;
|
||||||
|
|
||||||
|
if (scalar @props)
|
||||||
|
{
|
||||||
|
foreach (@props)
|
||||||
|
{
|
||||||
|
$db->print_prop($key, $_)
|
||||||
|
if defined $db->get_prop($key, $_);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$db->print_prop($key) ? exit 0 : exit 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_getprop
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
die "$commands{'getprop'}{'usage'}\n" unless $key;
|
||||||
|
|
||||||
|
my $prop = shift;
|
||||||
|
die "$commands{'getprop'}{'usage'}\n" unless scalar $prop;
|
||||||
|
|
||||||
|
my $val = $db->get_prop($key, $prop);
|
||||||
|
|
||||||
|
if (defined $val)
|
||||||
|
{
|
||||||
|
print "$val\n";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
exit 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_setprop
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
die "$commands{'setprop'}{'usage'}\n" unless $key;
|
||||||
|
die "$commands{'setprop'}{'usage'}\n" unless scalar @_;
|
||||||
|
die "$commands{'setprop'}{'usage'}\n" if scalar @_ % 2;
|
||||||
|
|
||||||
|
my %properties = @_;
|
||||||
|
|
||||||
|
foreach (sort keys %properties)
|
||||||
|
{
|
||||||
|
$db->set_prop($key, $_, $properties{$_});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DB_delprop
|
||||||
|
{
|
||||||
|
my $key = shift;
|
||||||
|
die "$commands{'delprop'}{'usage'}\n" unless $key;
|
||||||
|
die "$commands{'delprop'}{'usage'}\n" unless scalar @_;
|
||||||
|
|
||||||
|
foreach (@_)
|
||||||
|
{
|
||||||
|
$db->delete_prop($key, $_);
|
||||||
|
}
|
||||||
|
}
|
46
root/sbin/e-smith/expand-template
Normal file
46
root/sbin/e-smith/expand-template
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2004 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use Errno;
|
||||||
|
use Getopt::Long;
|
||||||
|
use esmith::templates;
|
||||||
|
|
||||||
|
my %options = ();
|
||||||
|
|
||||||
|
GetOptions(\%options, 'output_filename=s', 'expand_queue=s');
|
||||||
|
|
||||||
|
$options{'template_path'} = $ARGV[0] || die "Usage: $0 /path/to/file/to/expand\n";
|
||||||
|
|
||||||
|
$options{'output_filename'} = $options{'template_path'}
|
||||||
|
unless ( exists $options{'output_filename'} );
|
||||||
|
|
||||||
|
|
||||||
|
my %args = (
|
||||||
|
TEMPLATE_PATH => $options{'template_path'},
|
||||||
|
OUTPUT_FILENAME => $options{'output_filename'},
|
||||||
|
);
|
||||||
|
|
||||||
|
$args{TEMPLATE_EXPAND_QUEUE} = [$options{'expand_queue'}]
|
||||||
|
if exists $options{expand_queue};
|
||||||
|
|
||||||
|
if ( -f $options{'output_filename'} )
|
||||||
|
{
|
||||||
|
# If the target file exists, preserve its ownership and mode
|
||||||
|
use File::stat;
|
||||||
|
|
||||||
|
my $f = stat($options{'output_filename'} );
|
||||||
|
$args{UID} = $f->uid;
|
||||||
|
$args{GID} = $f->gid;
|
||||||
|
$args{PERMS} = $f->mode;
|
||||||
|
}
|
||||||
|
|
||||||
|
esmith::templates::processTemplate(\%args);
|
||||||
|
|
24
root/sbin/e-smith/signal-event
Normal file
24
root/sbin/e-smith/signal-event
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use Getopt::Long;
|
||||||
|
use esmith::event;
|
||||||
|
|
||||||
|
my $usage = "usage:
|
||||||
|
$0 eventname [arg1 [arg2...]]
|
||||||
|
";
|
||||||
|
|
||||||
|
my ($event, @args) = @ARGV;
|
||||||
|
die $usage unless $event;
|
||||||
|
|
||||||
|
my $exitcode = event_signal($event, @args);
|
||||||
|
|
||||||
|
# reverse exitcode for command-line usage
|
||||||
|
exit ($exitcode ? 0 : 1);
|
||||||
|
|
540
root/usr/share/perl5/vendor_perl/esmith/AccountsDB.pm
Normal file
540
root/usr/share/perl5/vendor_perl/esmith/AccountsDB.pm
Normal file
@ -0,0 +1,540 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::AccountsDB;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
use esmith::db;
|
||||||
|
|
||||||
|
use vars qw( $AUTOLOAD @ISA );
|
||||||
|
|
||||||
|
use esmith::DB::db;
|
||||||
|
@ISA = qw(esmith::DB::db);
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::AccountsDB - interface to esmith configuration database
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::AccountsDB;
|
||||||
|
my $a = esmith::AccountsDB->open;
|
||||||
|
|
||||||
|
my @users = $a->users();
|
||||||
|
my @groups = $a->groups();
|
||||||
|
my @ibays = $a->ibays();
|
||||||
|
my @printers = $a->printers();
|
||||||
|
my @pseudonyms = $a->pseudonyms();
|
||||||
|
|
||||||
|
$a->is_user_in_group($user, $group);
|
||||||
|
my @groups = $a->user_group_list($user);
|
||||||
|
$a->add_user_to_groups($user, @groups);
|
||||||
|
$a->remove_user_from_groups($user, @groups);
|
||||||
|
|
||||||
|
$a->create_user_auto_pseudonyms($user);
|
||||||
|
$a->remove_user_auto_pseudonyms($user);
|
||||||
|
$a->remove_all_user_pseudonyms($user);
|
||||||
|
my $dp = $a->dot_pseudonym($user);
|
||||||
|
my $up = $a->underbar_pseudonym($user);
|
||||||
|
my $uid = $a->get_next_uid();
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module provides an abstracted interface to the esmith accounts
|
||||||
|
database.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
our $VERSION = sprintf '%d.%03d', q$Revision: 1.18 $ =~ /: (\d+).(\d+)/;
|
||||||
|
|
||||||
|
=head2 open()
|
||||||
|
|
||||||
|
Loads an existing account database and returns an esmith::AccountsDB
|
||||||
|
object representing it.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use esmith::TestUtils qw(scratch_copy);
|
||||||
|
|
||||||
|
use_ok("esmith::AccountsDB");
|
||||||
|
use esmith::db;
|
||||||
|
use vars qw($a);
|
||||||
|
|
||||||
|
my $conf = scratch_copy('10e-smith-lib/accounts.conf');
|
||||||
|
$a = esmith::AccountsDB->open($conf);
|
||||||
|
isa_ok($a, 'esmith::AccountsDB');
|
||||||
|
is( $a->get("global")->prop('type'), "system", "We can get stuff from the db");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open {
|
||||||
|
my($class, $file) = @_;
|
||||||
|
$file = $file || $ENV{ESMITH_ACCOUNT_DB} || "accounts";
|
||||||
|
return $class->SUPER::open($file);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 open_ro()
|
||||||
|
|
||||||
|
Like esmith::DB->open_ro, but if given no $file it will try to open the
|
||||||
|
file in the ESMITH_ACCOUNT_DB environment variable or accounts.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open_ro {
|
||||||
|
my($class, $file) = @_;
|
||||||
|
$file = $file || $ENV{ESMITH_ACCOUNT_DB} || "accounts";
|
||||||
|
return $class->SUPER::open_ro($file);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 users(), groups(), ibays(), printers(), pseudonyms()
|
||||||
|
|
||||||
|
Returns a list of records (esmith::DB::db::Record objects) of the
|
||||||
|
given type.
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
foreach my $t (qw(users groups pseudonyms)) {
|
||||||
|
my @list = $a->$t();
|
||||||
|
ok(@list, "Got a list of $t");
|
||||||
|
isa_ok($list[0], 'esmith::DB::db::Record');
|
||||||
|
}
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub AUTOLOAD {
|
||||||
|
my $self = shift;
|
||||||
|
my ($called_sub_name) = ($AUTOLOAD =~ m/([^:]*)$/);
|
||||||
|
my @types = qw( users groups ibays printers pseudonyms);
|
||||||
|
if (grep /^$called_sub_name$/, @types) {
|
||||||
|
$called_sub_name =~ s/s$//g; # de-pluralize
|
||||||
|
return $self->get_all_by_prop(type => $called_sub_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=head1 GROUP MANAGEMENT
|
||||||
|
|
||||||
|
=head2 $a->is_user_in_group($user, $group)
|
||||||
|
|
||||||
|
Returns true if the user is a member of the group, false otherwise. The
|
||||||
|
arguments are a user name and a group name.
|
||||||
|
|
||||||
|
This routine will return undef if there is no such group, false (but
|
||||||
|
defined) if the user is not in the group, and true if the user is in the
|
||||||
|
group.
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
ok($a->is_user_in_group('bart', 'simpsons'), "Bart is in group Simpsons");
|
||||||
|
ok(not($a->is_user_in_group('moe', 'simpsons')), "Moe is not in group Simpsons");
|
||||||
|
ok(not(defined $a->is_user_in_group('moe', 'flanders')), "No such group as Flanders");
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub is_user_in_group {
|
||||||
|
my ($self, $user, $group) = @_;
|
||||||
|
$group = $self->get($group) || return undef;
|
||||||
|
my $members = $group->prop('Members');
|
||||||
|
|
||||||
|
return grep(/^$user$/, split /,/, $members) ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 $a->user_group_list($user)
|
||||||
|
|
||||||
|
Get a list of groups (by name) of which a user is a member. The $user argument
|
||||||
|
is simply the username.
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
my @groups = $a->user_group_list('bart');
|
||||||
|
is_deeply(\@groups, ['simpsons'], "Bart's group list is 'simpsons'");
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub user_group_list {
|
||||||
|
my ($self, $user) = @_;
|
||||||
|
my @groups = $self->groups();
|
||||||
|
my @user_groups;
|
||||||
|
foreach my $g (@groups) {
|
||||||
|
push(@user_groups, $g->key())
|
||||||
|
if $self->is_user_in_group($user, $g->key());
|
||||||
|
}
|
||||||
|
return @user_groups;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 $a->add_user_to_groups($user, @groups)
|
||||||
|
|
||||||
|
Given a list of groups (by name), adds the user to all of them.
|
||||||
|
|
||||||
|
Doesn't signal the group-modify event, just does the DB work.
|
||||||
|
|
||||||
|
Note: the method used here is a bit kludgy. It could result in a user
|
||||||
|
being in the same group twice.
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
my @groups = $a->groups();
|
||||||
|
$a->remove_user_from_groups('maggie', map { $_->key() } @groups);
|
||||||
|
my @mg = $a->user_group_list('maggie');
|
||||||
|
is(scalar @mg, 0, "Maggie has been removed from all groups");
|
||||||
|
$a->add_user_to_groups('maggie', 'simpsons');
|
||||||
|
@mg = $a->user_group_list('maggie');
|
||||||
|
is_deeply(\@mg, ['simpsons'], "Maggie has been added to group 'simpsons'");
|
||||||
|
$a->remove_user_from_groups('maggie', 'simpsons');
|
||||||
|
@mg = $a->user_group_list('maggie');
|
||||||
|
is_deeply(\@mg, [], "Maggie's been removed from all groups again");
|
||||||
|
$a->set_user_groups('maggie', 'simpsons');
|
||||||
|
@mg = $a->user_group_list('maggie');
|
||||||
|
is_deeply(\@mg, ['simpsons'], "Maggie's groups have been set to: 'simpsons'");
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub add_user_to_groups {
|
||||||
|
my ($self, $user, @groups) = @_;
|
||||||
|
GROUP: foreach my $group (@groups) {
|
||||||
|
unless (($group) = ($group =~ /(^[\w.-]+$)/))
|
||||||
|
{
|
||||||
|
warn "Group name doesn't look like a group!\n";
|
||||||
|
next GROUP;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $group_rec = $self->get($group) || next GROUP;
|
||||||
|
my @members = split(/,/, $group_rec->prop('Members'));
|
||||||
|
push @members, $user;
|
||||||
|
# Remove duplicates
|
||||||
|
my %members = map { $_ => 1 } @members;
|
||||||
|
$group_rec->set_prop('Members', join(',', sort keys %members));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 $a->remove_user_from_groups($user, @groups)
|
||||||
|
|
||||||
|
Given a list of groups, removes a user from all of them.
|
||||||
|
Doesn't signal the group-modify event, just does the DB work.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub remove_user_from_groups {
|
||||||
|
my ($self, $user, @groups) = @_;
|
||||||
|
|
||||||
|
GROUP: foreach my $g (@groups) {
|
||||||
|
my $group_rec = $self->get($g) || next GROUP;
|
||||||
|
my $members = $group_rec->prop('Members');
|
||||||
|
my @members = split (/,/, $members);
|
||||||
|
@members = grep (!/^$user$/, @members);
|
||||||
|
@members = qw(admin) unless @members; # admin *must* be in every group
|
||||||
|
$group_rec->set_prop('Members', join(',', @members));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 $a->set_user_groups($user, @groups)
|
||||||
|
|
||||||
|
Sets the user's groups in one fell swoop. Under the hood, it's removing
|
||||||
|
the user from every group they're in then adding them to the set you give.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub set_user_groups
|
||||||
|
{
|
||||||
|
my ($self, $user, @groups) = @_;
|
||||||
|
my @old_groups = $self->user_groups_list($user);
|
||||||
|
$self->remove_user_from_groups($user, @old_groups);
|
||||||
|
$self->add_user_to_groups($user, @groups);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head1 PSEUDONYM MANAGEMENT
|
||||||
|
|
||||||
|
=head2 $a->create_user_auto_pseudonyms($user)
|
||||||
|
|
||||||
|
Given a user name, creates standard pseudonyms ("dot" and "underbar" style)
|
||||||
|
for that user.
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
my $user = 'bart';
|
||||||
|
ok($a->pseudonyms(), "There are pseudonyms in the accounts db");
|
||||||
|
$a->remove_user_auto_pseudonyms($user);
|
||||||
|
ok(! $a->get('bart.simpson'), "Removed dot-pseudonym");
|
||||||
|
ok(! $a->get('bart_simpson'), "Removed underbar-pseudonym");
|
||||||
|
$a->create_user_auto_pseudonyms($user);
|
||||||
|
ok($a->get('bart.simpson'), "Created dot-pseudonym");
|
||||||
|
ok($a->get('bart_simpson'), "Created underbar-pseudonym");
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub create_user_auto_pseudonyms {
|
||||||
|
my ($self, $user) = @_;
|
||||||
|
my $user_rec = $self->get($user);
|
||||||
|
my $firstName = $user_rec->prop("FirstName");
|
||||||
|
my $lastName = $user_rec->prop("LastName");
|
||||||
|
|
||||||
|
my $dot_pseudonym = dot_pseudonym($self, $user);
|
||||||
|
my $underbar_pseudonym = underbar_pseudonym($self, $user);
|
||||||
|
|
||||||
|
my $dot_acct = $self->get($dot_pseudonym) ||
|
||||||
|
$self->new_record($dot_pseudonym, { type => 'pseudonym',
|
||||||
|
Account => $user} );
|
||||||
|
|
||||||
|
my $underbar_acct = $self->get($underbar_pseudonym) ||
|
||||||
|
$self->new_record($underbar_pseudonym, { type => 'pseudonym',
|
||||||
|
Account => $user} );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=head2 $a->remove_all_user_pseudonyms($user)
|
||||||
|
|
||||||
|
Given a username, remove any pseudonyms related to that user from the
|
||||||
|
accounts database. Also removes any pseudonyms related to a pseudonym
|
||||||
|
being removed. Returns the number of pseudonym records deleted.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub remove_all_user_pseudonyms {
|
||||||
|
my ($self, $user) = @_;
|
||||||
|
my $count = 0;
|
||||||
|
foreach my $p_rec (grep { $_->prop("Account") eq $user } $self->pseudonyms())
|
||||||
|
{
|
||||||
|
foreach my $p_p_rec (grep { $_->prop("Account") eq $p_rec->key } $self->pseudonyms())
|
||||||
|
{
|
||||||
|
$p_p_rec->delete;
|
||||||
|
$count++;
|
||||||
|
}
|
||||||
|
$p_rec->delete;
|
||||||
|
$count++;
|
||||||
|
}
|
||||||
|
return $count;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 $a->remove_user_auto_pseudonyms($user)
|
||||||
|
|
||||||
|
Given a username, remove the dot_pseudonym and underbar_pseudonym
|
||||||
|
related to that user from the accounts database. Returns the number
|
||||||
|
of pseudonym records deleted.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub remove_user_auto_pseudonyms {
|
||||||
|
my ($self, $user) = @_;
|
||||||
|
my $dot_pseudonym = dot_pseudonym($self, $user);
|
||||||
|
my $underbar_pseudonym = underbar_pseudonym($self, $user);
|
||||||
|
my $count = 0;
|
||||||
|
foreach my $p_rec ($self->get($dot_pseudonym),
|
||||||
|
$self->get($underbar_pseudonym))
|
||||||
|
{
|
||||||
|
if (defined $p_rec && $p_rec->prop("type") eq "pseudonym" &&
|
||||||
|
$p_rec->prop("Account") eq $user)
|
||||||
|
{
|
||||||
|
$p_rec->delete;
|
||||||
|
$count++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $count;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 $a->dot_pseudonym($user)
|
||||||
|
|
||||||
|
Returns the "dot"-style pseudonym for a user as a string. For instance,
|
||||||
|
dot_pseudonym("bart") might return "bart.simpson".
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub dot_pseudonym {
|
||||||
|
my ($self, $user) = @_;
|
||||||
|
my $user_rec = $self->get($user);
|
||||||
|
my $firstName = $user_rec->prop("FirstName");
|
||||||
|
my $lastName = $user_rec->prop("LastName");
|
||||||
|
|
||||||
|
my $dot_pseudonym = lc("$firstName $lastName");
|
||||||
|
|
||||||
|
$dot_pseudonym =~ s/^\s+//; # Strip leading whitespace
|
||||||
|
$dot_pseudonym =~ s/\s+$//; # Strip trailing whitespace
|
||||||
|
$dot_pseudonym =~ s/\s+/ /g; # Multiple spaces become single spaces
|
||||||
|
$dot_pseudonym =~ s/\s/./g; # Change all spaces to dots
|
||||||
|
return $dot_pseudonym;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 $a->underbar_pseudonym($user)
|
||||||
|
|
||||||
|
Returns the "underbar"-style pseudonym for a user as a string. For instance,
|
||||||
|
underbar_pseudonym("bart") might return "bart_simpson".
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my @users = $a->users();
|
||||||
|
my $user = 'bart';
|
||||||
|
my $rec = $a->get($user);
|
||||||
|
my $firstName = $rec->prop("FirstName");
|
||||||
|
my $lastName = $rec->prop("LastName");
|
||||||
|
my $up = $a->underbar_pseudonym($user);
|
||||||
|
is($up, "bart_simpson", "Underbar pseudonym created correctly");
|
||||||
|
my $dp = $a->dot_pseudonym($user);
|
||||||
|
is($dp, "bart.simpson", "Underbar pseudonym created correctly");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub underbar_pseudonym {
|
||||||
|
my ($self, $user) = @_;
|
||||||
|
my $user_rec = $self->get($user);
|
||||||
|
my $firstName = $user_rec->prop("FirstName");
|
||||||
|
my $lastName = $user_rec->prop("LastName");
|
||||||
|
|
||||||
|
my $underbar_pseudonym = lc("$firstName $lastName");
|
||||||
|
|
||||||
|
$underbar_pseudonym =~ s/^\s+//; # Strip leading whitespace
|
||||||
|
$underbar_pseudonym =~ s/\s+$//; # Strip trailing whitespace
|
||||||
|
$underbar_pseudonym =~ s/\s+/ /g; # Multiple spaces become single spaces
|
||||||
|
$underbar_pseudonym =~ s/\s/_/g; # Change all spaces to underbars
|
||||||
|
return $underbar_pseudonym;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 $a->activeUsers()
|
||||||
|
|
||||||
|
Returns the number of active users, ie, accounts which have passwords set and
|
||||||
|
are of type 'user'.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $numActiveUsers = scalar $a->activeUsers();
|
||||||
|
like($numActiveUsers, qr/[0-9]+/, "active users returns a number");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub activeUsers()
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my @users = $self->users();
|
||||||
|
|
||||||
|
return unless @users;
|
||||||
|
return grep { $_->prop("PasswordSet") eq 'yes' } @users;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 get_next_uid
|
||||||
|
|
||||||
|
Returns the next available UID from /etc/passwd. All UIDs are above the range
|
||||||
|
reserved for 'system' accounts (currently 5000).
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
SKIP: {
|
||||||
|
skip "Must be root to run get_next_uid" if $<;
|
||||||
|
my $u = $a->get_next_uid();
|
||||||
|
ok($u > 5000, "UID should be greater than 5000");
|
||||||
|
ok(! getpwuid($u), "UID should not yet exist");
|
||||||
|
}
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get_next_uid {
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
|
||||||
|
my $id;
|
||||||
|
my $db = esmith::ConfigDB->open || die "Couldn't open config db";
|
||||||
|
|
||||||
|
if ($id = $db->get('MinUid'))
|
||||||
|
{
|
||||||
|
$id = $id->value();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$db->new_record('MinUid');
|
||||||
|
$id = 5000;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $maxid = 1 << 31;
|
||||||
|
setpwent();
|
||||||
|
setgrent();
|
||||||
|
while (getpwuid $id || getgrgid $id)
|
||||||
|
{
|
||||||
|
die "All userids in use" if ($id == $maxid);
|
||||||
|
$id++;
|
||||||
|
}
|
||||||
|
endpwent();
|
||||||
|
endgrent();
|
||||||
|
|
||||||
|
$db->set_value('MinUid', $id + 1);
|
||||||
|
|
||||||
|
return $id;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 new_record ($key, \%props)
|
||||||
|
|
||||||
|
This method is overridden from esmith::DB::db. We do an additional check
|
||||||
|
for implicit accounts here - accounts that exist in /etc/passwd but not
|
||||||
|
in the db. Otherwise it behaves just like the superclass method.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
isnt($a->new_record("root", {type=>'system'}), "OK",
|
||||||
|
"can't create existing account");
|
||||||
|
is($a->get("nobody"), undef, "nobody doesn't exist in db");
|
||||||
|
isnt($a->new_record("nobody", {type=>'system'}), "OK",
|
||||||
|
"can't create account in /etc/passwd");
|
||||||
|
isnt($a->new_record("screwy", {type=>'user'}), undef,
|
||||||
|
"created a regular user");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub new_record
|
||||||
|
{
|
||||||
|
my ($self, $key, $props) = @_;
|
||||||
|
|
||||||
|
if(getpwnam($key) || getgrnam($key))
|
||||||
|
{
|
||||||
|
warn "Attempt to create account '$key' which already exists ",
|
||||||
|
"in passwd";
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
return $self->SUPER::new_record($key, $props);
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 validate_account_name ($name)
|
||||||
|
|
||||||
|
Check $name to see if it is a valid account name. Valid account names
|
||||||
|
start with a letter or number and contain only letters, numbers,
|
||||||
|
underscores, dots and dashes.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
is($a->validate_account_name("root"), "OK", "root is a valid name");
|
||||||
|
is($a->validate_account_name("fred.frog"), "OK", "fred.frog is a valid name");
|
||||||
|
is($a->validate_account_name("jane_doe"), "OK", "jane_doe is a valid name");
|
||||||
|
isnt($a->validate_account_name("^root"), "OK", "^root is not a valid name");
|
||||||
|
is(esmith::AccountsDB::validate_account_name("root"), "OK", "called as a function");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub validate_account_name
|
||||||
|
{
|
||||||
|
my $self;
|
||||||
|
# Were we called as a method or a function?
|
||||||
|
if($#_ > 0)
|
||||||
|
{
|
||||||
|
$self = shift;
|
||||||
|
}
|
||||||
|
my $name = shift;
|
||||||
|
return ($name =~ /[^0-9a-z\-_\.]/ or $name !~ /^[a-z]/) ? undef : 'OK';
|
||||||
|
}
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
SME Server Developers <bugs@e-smith.com>
|
||||||
|
|
||||||
|
See http://www.e-smith.org/ for more information
|
||||||
|
|
||||||
|
|
||||||
|
|
398
root/usr/share/perl5/vendor_perl/esmith/ConfigDB.pm
Normal file
398
root/usr/share/perl5/vendor_perl/esmith/ConfigDB.pm
Normal file
@ -0,0 +1,398 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::ConfigDB;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
|
||||||
|
use vars qw( $AUTOLOAD @ISA );
|
||||||
|
|
||||||
|
use esmith::DB::db;
|
||||||
|
@ISA = qw( esmith::DB::db );
|
||||||
|
|
||||||
|
use esmith::ConfigDB::Record;
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::ConfigDB - interface to esmith configuration database
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
my $db = esmith::ConfigDB->open;
|
||||||
|
my $db = esmith::ConfigDB->open_ro;
|
||||||
|
|
||||||
|
my @services = $db->services();
|
||||||
|
|
||||||
|
# Singleton Records
|
||||||
|
my $record = $db->get($key);
|
||||||
|
my $value = $record->value;
|
||||||
|
$record->set_value($value);
|
||||||
|
|
||||||
|
# BAD!
|
||||||
|
my $value = $db->get($key)->value() # Throws a runtime error if $key
|
||||||
|
# doesn't exist
|
||||||
|
$value = $db->get($key)->prop($p) # Throws a runtime error if $key
|
||||||
|
# doesn't exist
|
||||||
|
|
||||||
|
# GOOD
|
||||||
|
my $record = $db->get($key);
|
||||||
|
my $value;
|
||||||
|
if ($record)
|
||||||
|
{
|
||||||
|
$value = $record->prop($prop);
|
||||||
|
}
|
||||||
|
|
||||||
|
# Typed Records (eventually they all will be)
|
||||||
|
my $prop = $record->prop($p);
|
||||||
|
$record->set_prop($prop, $propvalue);
|
||||||
|
my $value = $db->get_value($key) # Returns undef if record doesn't exist
|
||||||
|
$value = $db->get_prop($key, $p) # Returns undef if record doesn't exist
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module provides an abstracted interface to the esmith master
|
||||||
|
configuration database.
|
||||||
|
|
||||||
|
Unless otherwise noted, esmith::ConfigDB acts like esmith::DB::db.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
our $VERSION = sprintf '%d.%03d', q$Revision: 1.29 $ =~ /: (\d+).(\d+)/;
|
||||||
|
|
||||||
|
=head2 open()
|
||||||
|
|
||||||
|
Like esmith::DB->open, but if given no $file it will try to open the
|
||||||
|
file in the ESMITH_CONFIG_DB environment variable or configuration.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use esmith::TestUtils qw(scratch_copy);
|
||||||
|
use_ok("esmith::ConfigDB");
|
||||||
|
|
||||||
|
my $scratch_copy_of_conf = scratch_copy('10e-smith-lib/configuration.conf');
|
||||||
|
$C = esmith::ConfigDB->open($scratch_copy_of_conf);
|
||||||
|
isa_ok($C, 'esmith::ConfigDB');
|
||||||
|
is( $C->get("AccessType")->prop('type'), "dedicated",
|
||||||
|
"We can get stuff from the db");
|
||||||
|
|
||||||
|
is( $C->get_prop("AccessType",'type'), "dedicated",
|
||||||
|
"We can get_prop stuff from the db");
|
||||||
|
|
||||||
|
is( $C->get_value("AccessType"), "dedicated",
|
||||||
|
"We can get_value stuff from the db");
|
||||||
|
|
||||||
|
is( $C->get_value("NoSuchKey"), undef,
|
||||||
|
"We can get_value non-existant keys");
|
||||||
|
|
||||||
|
is( $C->get_prop("diald","status"), "disabled",
|
||||||
|
"We can get_prop stuff from the db");
|
||||||
|
|
||||||
|
is( $C->get_prop("NoSuchKey","NoSuchProp"), undef,
|
||||||
|
"We can get_prop non-existant keys");
|
||||||
|
|
||||||
|
is( $C->get_prop("diald","NoSuchProp"), undef,
|
||||||
|
"We can get_prop non-existant props");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
$file = $file || $ENV{ESMITH_CONFIG_DB} || "configuration";
|
||||||
|
return $class->SUPER::open($file);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 open_ro()
|
||||||
|
|
||||||
|
Like esmith::DB->open_ro, but if given no $file it will try to open the
|
||||||
|
file in the ESMITH_CONFIG_DB environment variable or configuration.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open_ro
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
$file = $file || $ENV{ESMITH_CONFIG_DB} || "configuration";
|
||||||
|
return $class->SUPER::open_ro($file);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 new_record()
|
||||||
|
|
||||||
|
This method creates a new record in the configuration database. As arguments,
|
||||||
|
it expects the key to the record, followed by a hash references with its
|
||||||
|
properties, including the type.
|
||||||
|
|
||||||
|
my $db = esmith::ConfigDB->open;
|
||||||
|
my $record = $db->new_record('zope', { type => 'service',
|
||||||
|
status => 'disabled' });
|
||||||
|
|
||||||
|
my %defaults = qw(
|
||||||
|
type => 'service',
|
||||||
|
status => 'disabled',
|
||||||
|
maintainer => 'admin@domain.com'
|
||||||
|
);
|
||||||
|
my $record = $db->get('zope');
|
||||||
|
unless ($record)
|
||||||
|
{
|
||||||
|
$record = $db->new_record('zope', \%defaults);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 get()
|
||||||
|
|
||||||
|
Like their esmith::DB counterparts except they return
|
||||||
|
esmith::ConfigDB::Record objects which have a few extra methods.
|
||||||
|
|
||||||
|
my $record = $db->get('zope');
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $rec = eval { $C->get("I_dont_exist"); };
|
||||||
|
ok( !$rec, 'get() on a non-existent key' );
|
||||||
|
is( $@, '', ' doesnt blow up' );
|
||||||
|
|
||||||
|
isa_ok( $C->get("AccessType"), 'esmith::ConfigDB::Record',
|
||||||
|
"get()ened records are ConfigDB::Records" );
|
||||||
|
|
||||||
|
$rec = $C->new_record("I_dont_exist", { type => "foo" });
|
||||||
|
isa_ok( $rec, 'esmith::ConfigDB::Record',
|
||||||
|
"new_record()s are ConfigDB::Records" );
|
||||||
|
$rec->delete("I_dont_exist");
|
||||||
|
ok( !$C->get("I_dont_exist"), 'delete()' );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get
|
||||||
|
{
|
||||||
|
my ($self) = shift;
|
||||||
|
|
||||||
|
my $rec = $self->SUPER::get(@_);
|
||||||
|
return $rec ? bless $rec, 'esmith::ConfigDB::Record' : undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub new_record
|
||||||
|
{
|
||||||
|
my ($self) = shift;
|
||||||
|
|
||||||
|
my $rec = $self->SUPER::new_record(@_);
|
||||||
|
return $rec ? bless $rec, 'esmith::ConfigDB::Record' : undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 getLocale()
|
||||||
|
|
||||||
|
Retrieves the locale and keyboard settings from the configuration database.
|
||||||
|
Returns ($lang, $kbdtype, $keytable) on success. Returns undef if the record
|
||||||
|
doesn't exist.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub getLocale
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $rec = $self->get('sysconfig') or return undef;
|
||||||
|
my $lang = $rec->prop('Language') || 'en_US';
|
||||||
|
my $kbdtype = $rec->prop('KeyboardType') || 'pc';
|
||||||
|
my $keytable = $rec->prop('Keytable') || 'us';
|
||||||
|
return ( $lang, $kbdtype, $keytable );
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 hosts_allow_spec ($service [,$daemon])
|
||||||
|
|
||||||
|
Given a service, return the string suitable for /etc/hosts.allow,
|
||||||
|
checking to see if the service is defined, whether it is enabled and
|
||||||
|
whether access is set to public, private, or localhost.
|
||||||
|
|
||||||
|
An optional argument provides the tag which appears in hosts.allow. If not
|
||||||
|
given, the service name is used.
|
||||||
|
|
||||||
|
For example, one of the following:
|
||||||
|
|
||||||
|
# 'oidentd' is not defined in the configuration database
|
||||||
|
# 'oidentd' is disabled in the configuration database
|
||||||
|
in.identd: 127.0.0.1
|
||||||
|
in.identd: 127.0.0.1 192.168.1.1/255.255.255.0
|
||||||
|
in.identd: ALL
|
||||||
|
|
||||||
|
And here's the hosts.allow fragment:
|
||||||
|
|
||||||
|
{
|
||||||
|
$OUT = $DB->hosts_allow_spec('oidentd', 'in.identd');
|
||||||
|
}
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub hosts_allow_spec
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $service_name = shift;
|
||||||
|
my $daemon = shift || $service_name;
|
||||||
|
|
||||||
|
my $service = $self->get($service_name)
|
||||||
|
or return
|
||||||
|
"# '$service_name' is not defined in the configuration database";
|
||||||
|
|
||||||
|
my $status = $service->prop('status') || "disabled";
|
||||||
|
return "# '$service_name' is disabled in the configuration database"
|
||||||
|
unless ( $status eq "enabled" );
|
||||||
|
|
||||||
|
my $access = $service->prop('access') || "private";
|
||||||
|
|
||||||
|
use esmith::NetworksDB;
|
||||||
|
my $ndb = esmith::NetworksDB->open_ro;
|
||||||
|
|
||||||
|
my @spec = ( "$daemon:", $ndb->local_access_spec($access) );
|
||||||
|
return "@spec";
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 wins_server
|
||||||
|
|
||||||
|
Return the value of the WINS server from the config db
|
||||||
|
or undef if we don't have a WINS server set and we are
|
||||||
|
not the domain master
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub wins_server
|
||||||
|
{
|
||||||
|
my ($self) = @_;
|
||||||
|
|
||||||
|
my $wins_server = $self->get_prop( 'smb', 'WINSServer' );
|
||||||
|
|
||||||
|
return $wins_server if $wins_server;
|
||||||
|
|
||||||
|
my $server_role = $self->get_prop( 'smb', 'ServerRole' ) || 'WS';
|
||||||
|
|
||||||
|
return $self->get_prop( 'InternalInterface', 'IPAddress' )
|
||||||
|
if $server_role =~ m{^(PDC|ADS)$};
|
||||||
|
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 services()
|
||||||
|
|
||||||
|
Returns a list of services in the configuration database
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
foreach my $t (qw(services)) {
|
||||||
|
my @list = $C->$t();
|
||||||
|
ok(@list, "Got a list of $t");
|
||||||
|
}
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub AUTOLOAD
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my ($called_sub_name) = ( $AUTOLOAD =~ m/([^:]*)$/ );
|
||||||
|
my @types = qw( services );
|
||||||
|
if ( grep /^$called_sub_name$/, @types )
|
||||||
|
{
|
||||||
|
$called_sub_name =~ s/s$//g; # de-pluralize
|
||||||
|
return $self->list_by_type($called_sub_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 _loadDefaults ($forceReset)
|
||||||
|
|
||||||
|
Behaves just like the esmith::DB method of the same name. This is a private
|
||||||
|
method used internally.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$scratch_copy_of_conf = scratch_copy('10e-smith-lib/configuration.conf', 'configuration.conf.scratch');
|
||||||
|
is ($scratch_copy_of_conf, 'configuration.conf.scratch', 'scratch copy name');
|
||||||
|
$ENV{'ESMITH_CONFIG_DB'} = $scratch_copy_of_conf;
|
||||||
|
$C = esmith::ConfigDB->open();
|
||||||
|
is ($C->{file}, $scratch_copy_of_conf, 'file name');
|
||||||
|
my $accesstype = $C->get("AccessType")->value;
|
||||||
|
ok($C->new_record('foobar', {type=>'service', status=>'disabled'}),
|
||||||
|
"Set up foobar record");
|
||||||
|
is($C->get('foobar')->prop('status'), "disabled", "foobar is disabled");
|
||||||
|
$ENV{'ESMITH_DB_DEFAULTSDIR'} = "10e-smith-lib/db";
|
||||||
|
ok($C->_loadDefaults(), "Loaded defaults");
|
||||||
|
is($C->get('foobar')->prop('status'), 'enabled', "We forced status enabled");
|
||||||
|
is($C->get('bazbar')->prop('status'), 'enabled', "We included a new default");
|
||||||
|
is($C->get('AccessType')->value, $accesstype, "AccessType has not changed");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=head2 record_has_defaults ($name)
|
||||||
|
|
||||||
|
Behaves just like the esmith::DB method of the same name.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$ENV{'ESMITH_DB_DEFAULTSDIR'} = "10e-smith-lib/db";
|
||||||
|
is($C->record_has_defaults('foobar'), 1, "foobar has some defaults");
|
||||||
|
is($C->record_has_defaults('notthisone'), undef, "notthisone does not");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
# There would normally be a method here, but we inherit _loadDefaults
|
||||||
|
# from esmith::DB. The tests need to go here because the superclass is
|
||||||
|
# all virtual and testing requires concrete open/get/set methods.
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 migrate
|
||||||
|
|
||||||
|
Just like the esmith::DB method of the same name.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$scratch_copy_of_conf = scratch_copy('10e-smith-lib/configuration.conf', 'configuration.conf.scratch');
|
||||||
|
is ($scratch_copy_of_conf, 'configuration.conf.scratch', 'scratch copy name');
|
||||||
|
$ENV{'ESMITH_CONFIG_DB'} = $scratch_copy_of_conf;
|
||||||
|
$C = esmith::ConfigDB->open();
|
||||||
|
is($C->get('quux'), undef, "No quux here");
|
||||||
|
$ENV{'ESMITH_DB_DEFAULTSDIR'} = "10e-smith-lib/db";
|
||||||
|
ok($C->migrate(), "We can migrate");
|
||||||
|
my $quux = $C->get('quux');
|
||||||
|
ok($quux, "We got quux");
|
||||||
|
is($quux->prop('status'), 'enabled', "We migrated to quux");
|
||||||
|
$quux->delete;
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
# There would normally be a method here, but we inherit migrate
|
||||||
|
# from esmith::DB. The tests need to go here because the superclass is
|
||||||
|
# all virtual and testing requires concrete open/get/set methods.
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
SME Server Developers <bugs@e-smith.com>
|
||||||
|
|
||||||
|
=head1 SEE ALSO
|
||||||
|
|
||||||
|
L<esmith::DB>, L<esmith::DB::db>, L<esmith::AccountsDB>, L<esmith::DomainsDB>,
|
||||||
|
L<esmith::HostsDB>, L<esmith::NetworksDB>, L<esmith::ConfigDB::Record>
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
108
root/usr/share/perl5/vendor_perl/esmith/ConfigDB/Record.pm
Normal file
108
root/usr/share/perl5/vendor_perl/esmith/ConfigDB/Record.pm
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::ConfigDB::Record;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
require esmith::DB::db::Record;
|
||||||
|
our @ISA = qw(esmith::DB::db::Record);
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::ConfigDB::Record - record in an esmith::ConfigDB database.
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
Just like esmith::DB::db::Record except...
|
||||||
|
|
||||||
|
my $value = $record->value;
|
||||||
|
$record->set_value($value);
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This provides some extra functionality needed by the esmith::ConfigDB
|
||||||
|
databases.
|
||||||
|
|
||||||
|
Unless noted, it works just like esmith::DB::db::Record.
|
||||||
|
|
||||||
|
=head2 New Methods
|
||||||
|
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item B<value>
|
||||||
|
|
||||||
|
=item B<set_value>
|
||||||
|
|
||||||
|
my $value = $record->value;
|
||||||
|
$record->set_value($value);
|
||||||
|
|
||||||
|
Gets/sets the value of the $record. Some ConfigDB entries don't have
|
||||||
|
a set of properties, but rather a single value.
|
||||||
|
|
||||||
|
It will warn if you use these on $records with properties.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
$Scratch_Conf = '10e-smith-lib/scratch.conf';
|
||||||
|
unlink $Scratch_Conf;
|
||||||
|
$c = esmith::ConfigDB->create($Scratch_Conf);
|
||||||
|
END { unlink $Scratch_Conf }
|
||||||
|
|
||||||
|
{
|
||||||
|
my $warning = '';
|
||||||
|
local $SIG{__WARN__} = sub { $warning = join '', @_ };
|
||||||
|
|
||||||
|
my $wib = $c->new_record('wibble', { type => 'yar' });
|
||||||
|
$wib->set_value('foo');
|
||||||
|
is( $wib->value, 'foo', 'value/set_value' );
|
||||||
|
is( $warning, '', ' no warning' );
|
||||||
|
|
||||||
|
$wib->set_prop(bar => 42);
|
||||||
|
is( $wib->value, 'foo' );
|
||||||
|
like( $warning, qr/value\(\) should not be used on records with properties, use prop\(\)/, 'value() warns if the record has props');
|
||||||
|
|
||||||
|
$wib->set_value(92);
|
||||||
|
like( $warning, qr/set_value\(\) should not be used on records with properties, use set_prop\(\)/, 'value() warns if the record has props');
|
||||||
|
is( $wib->value, 92 );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub value {
|
||||||
|
my($self) = shift;
|
||||||
|
|
||||||
|
my %props = $self->props;
|
||||||
|
warn "value() should not be used on records with properties, use prop()" if
|
||||||
|
keys %props > 1;
|
||||||
|
return $self->prop('type');
|
||||||
|
}
|
||||||
|
|
||||||
|
sub set_value {
|
||||||
|
my($self, $value) = @_;
|
||||||
|
|
||||||
|
my %props = $self->props;
|
||||||
|
warn "set_value() should not be used on records with properties, ".
|
||||||
|
"use set_prop()" if keys %props > 1;
|
||||||
|
return $self->set_prop('type', $value);
|
||||||
|
}
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head1 SEE ALSO
|
||||||
|
|
||||||
|
L<esmith::ConfigDB>, L<esmith::DB::db::Record>
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
22
root/usr/share/perl5/vendor_perl/esmith/ConfigDB/UTF8.pm
Normal file
22
root/usr/share/perl5/vendor_perl/esmith/ConfigDB/UTF8.pm
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2008 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::ConfigDB::UTF8;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
|
||||||
|
use esmith::DB::db;
|
||||||
|
use esmith::config::utf8;
|
||||||
|
our @ISA = qw( esmith::DB::db );
|
||||||
|
|
||||||
|
sub tie_class
|
||||||
|
{
|
||||||
|
return 'esmith::config::utf8';
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
||||||
|
|
705
root/usr/share/perl5/vendor_perl/esmith/DB.pm
Normal file
705
root/usr/share/perl5/vendor_perl/esmith/DB.pm
Normal file
@ -0,0 +1,705 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::DB;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
use Carp;
|
||||||
|
use File::Basename;
|
||||||
|
use esmith::templates;
|
||||||
|
|
||||||
|
use constant TRUE => 1;
|
||||||
|
use constant FALSE => 0;
|
||||||
|
|
||||||
|
our $VERSION = sprintf '%d.%03d', q$Revision: 1.40 $ =~ /: (\d+).(\d+)/;
|
||||||
|
our $Error = undef;
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::DB - virtual interface to E-Smith databases
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
# Note: Do not instantiate this class directly. Use a subclass.
|
||||||
|
# Examples in this documentation where you see it being used directly
|
||||||
|
# are merely for consistency. Substitute a subclass in our examples.
|
||||||
|
use esmith::DB;
|
||||||
|
my $db = esmith::DB->create($filename) or
|
||||||
|
die esmith::DB->error;
|
||||||
|
my $db = esmith::DB->open($filename) or
|
||||||
|
die esmith::DB->error;
|
||||||
|
my $db = esmith::DB->open_ro($filename) or
|
||||||
|
die esmith::DB->error;
|
||||||
|
|
||||||
|
my %DB = esmith::DB->as_hash($filename) or
|
||||||
|
die esmith::DB->error;
|
||||||
|
|
||||||
|
$db->reload;
|
||||||
|
|
||||||
|
my $file = $db->file;
|
||||||
|
|
||||||
|
my $record = $db->new_record($key, \%properties);
|
||||||
|
|
||||||
|
my $record = $db->get($key);
|
||||||
|
my @all_records = $db->get_all;
|
||||||
|
my @all_records_by_prop = $db->get_all_by_prop($prop => $val);
|
||||||
|
|
||||||
|
$db->set_prop($key, $prop, $value);
|
||||||
|
$db->set_value($key, $value);
|
||||||
|
|
||||||
|
$db->set_prop($key, $prop, $value, type => $type);
|
||||||
|
$db->set_value($key, $value, create => 0);
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module is a general interface to E-Smith's databases of various
|
||||||
|
types and formats. It is not intended to be used directly, but that
|
||||||
|
subclasses will implement the interface presented here to provide a
|
||||||
|
single interface no matter what the underlying format.
|
||||||
|
|
||||||
|
For example, there is esmith::DB::db to interface with esmith::db
|
||||||
|
flatfile databases. There could also be esmith::DB::Berkeley to use
|
||||||
|
Berkeley database files, or even esmith::DB::DBI.
|
||||||
|
|
||||||
|
Most of the methods herein are "virtual". They don't exist. The
|
||||||
|
subclass is responsible for impelmenting them. There are a handful of
|
||||||
|
concrete methods that have been implemented for you that should work
|
||||||
|
with any subclass.
|
||||||
|
|
||||||
|
=head2 Virtual Methods
|
||||||
|
|
||||||
|
This is the esmith::DB interface. Subclassers are expected to
|
||||||
|
implement these methods.
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item I<create>
|
||||||
|
|
||||||
|
my $db = esmith::DB->create($new_config_file) ||
|
||||||
|
die esmith::DB->error;
|
||||||
|
|
||||||
|
Creates a brand new, empty configuration database and returns a
|
||||||
|
subclass of the esmith::DB object representing it.
|
||||||
|
|
||||||
|
Should the $new_config_file already exist or for some reason you can't
|
||||||
|
write to it, esmith::DB->error will return the reason and
|
||||||
|
create() will return false.
|
||||||
|
|
||||||
|
=item I<open>
|
||||||
|
|
||||||
|
my $db = esmith::DB->open($config_file) ||
|
||||||
|
die esmith::DB->error
|
||||||
|
|
||||||
|
Loads an existing configuration database and returns a
|
||||||
|
subclass of the esmith::DB::db object representing it.
|
||||||
|
|
||||||
|
Should the $config_file not exist or not be openable it will return
|
||||||
|
false and esmith::DB->error will contain the reason.
|
||||||
|
|
||||||
|
=item I<open_ro>
|
||||||
|
|
||||||
|
my $db = esmith::DB->open_ro($config_file) ||
|
||||||
|
die esmith::DB->error;
|
||||||
|
|
||||||
|
Like open(), but the database is read-only. new_record() and all methods
|
||||||
|
which could change a record (set_prop(), merge_props(), delete(), etc...)
|
||||||
|
will both throw exceptions if used.
|
||||||
|
|
||||||
|
=item I<error>
|
||||||
|
|
||||||
|
my $error = esmith::DB->error;
|
||||||
|
|
||||||
|
Returns a string describing the error from the last failing method.
|
||||||
|
|
||||||
|
=item I<reload>
|
||||||
|
|
||||||
|
$db->reload;
|
||||||
|
|
||||||
|
Flushes out the $db's cache (if there is one) and reloads all
|
||||||
|
configuration data from disk.
|
||||||
|
|
||||||
|
=item I<file>
|
||||||
|
|
||||||
|
my $file = $db->file;
|
||||||
|
|
||||||
|
File which this $db represents.
|
||||||
|
|
||||||
|
=item I<new_record>
|
||||||
|
|
||||||
|
my $record = $db->new_record($key, \%properties);
|
||||||
|
|
||||||
|
Adds a new record at $key in the $db setting it to the given
|
||||||
|
%properties. Returns a subclass of the esmith::DB::Record object.
|
||||||
|
|
||||||
|
If a record already exists for the $key it will return false.
|
||||||
|
|
||||||
|
=item I<get>
|
||||||
|
|
||||||
|
my $record = $db->get($key);
|
||||||
|
|
||||||
|
Gets an existing record from the $db with the given $key. Returns an
|
||||||
|
esmith::DB::Record object representing the data in $key.
|
||||||
|
|
||||||
|
If there's no record for the $key it will return false.
|
||||||
|
|
||||||
|
=item I<get_all>
|
||||||
|
|
||||||
|
my @records = $db->get_all;
|
||||||
|
|
||||||
|
Gets all the records out of the given $db as a list of
|
||||||
|
esmith::DB::Record objects.
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head2 Concrete methods
|
||||||
|
|
||||||
|
These are all implemented in terms of the esmith::DB interface and
|
||||||
|
its not necessary for a subclass to implement them.
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item I<as_hash>
|
||||||
|
|
||||||
|
my %db = $db->as_hash;
|
||||||
|
my %db = esmith::DB->as_hash($file);
|
||||||
|
|
||||||
|
Returns the entire database as a hash of hashes. Each key is a key in
|
||||||
|
the database, and the value is a hash of it's properties.
|
||||||
|
|
||||||
|
my $value = $db{some_key}{some_prop};
|
||||||
|
|
||||||
|
When used as an object method it will use the already opened database.
|
||||||
|
When used as a class method it will open the given $file.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub as_hash
|
||||||
|
{
|
||||||
|
my ( $proto, $file ) = @_;
|
||||||
|
my $class = ref $proto || $proto;
|
||||||
|
|
||||||
|
my $self;
|
||||||
|
if ( ref $proto )
|
||||||
|
{ # object method
|
||||||
|
$self = $proto;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{ # class method
|
||||||
|
$self = $class->open($file) or return;
|
||||||
|
}
|
||||||
|
my %hash = ();
|
||||||
|
foreach my $rec ( $self->get_all )
|
||||||
|
{
|
||||||
|
my $key = $rec->key;
|
||||||
|
my %props = $rec->props;
|
||||||
|
|
||||||
|
# Setup the hash
|
||||||
|
$hash{$key} = \%props;
|
||||||
|
}
|
||||||
|
|
||||||
|
return %hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<get_all_by_prop>
|
||||||
|
|
||||||
|
my @records_by_prop = $db->get_all_by_prop($property => $value);
|
||||||
|
|
||||||
|
Like get_all() except it gets only those records whose $property has
|
||||||
|
the given $value. For properties with multiple comma-delimited values
|
||||||
|
(ie: name|val1,val2,val3), only one of the properties needs to match.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get_all_by_prop
|
||||||
|
{
|
||||||
|
my ( $self, $prop, @rest ) = @_;
|
||||||
|
my %props;
|
||||||
|
|
||||||
|
if ( ref($prop) eq 'HASH' )
|
||||||
|
{
|
||||||
|
carp "get_all_by_prop called with anonymous hash argument";
|
||||||
|
%props = ( %{$prop} );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
%props = ($prop, @rest);
|
||||||
|
}
|
||||||
|
my @things = sort { $a->key cmp $b->key } grep
|
||||||
|
{
|
||||||
|
my $found = 1;
|
||||||
|
while ( my ($p, $v) = each (%props) )
|
||||||
|
{
|
||||||
|
$found &= ( defined $_->prop($p) and $_->prop($p) =~ /(?:^|,)$v(?:,|$)/ );
|
||||||
|
}
|
||||||
|
$found;
|
||||||
|
} $self->get_all;
|
||||||
|
@things;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<get_value>
|
||||||
|
|
||||||
|
my $value = $db->get_value( $key );
|
||||||
|
|
||||||
|
Shortcut method to get the value from the record defined by the given
|
||||||
|
key. Returns undef if the record does not exist.
|
||||||
|
|
||||||
|
The following code is unsafe if the key doesn't exist:
|
||||||
|
|
||||||
|
my $value = $db->get("foo")->value || 'default';
|
||||||
|
|
||||||
|
and should be:
|
||||||
|
|
||||||
|
my $value = 'default';
|
||||||
|
|
||||||
|
if (my $r = $db->get("foo"))
|
||||||
|
{
|
||||||
|
$value = $r->value;
|
||||||
|
}
|
||||||
|
|
||||||
|
With this method, you can use:
|
||||||
|
|
||||||
|
my $value = $db->get_value("foo") || 'default';
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get_value
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $item = $self->get(shift);
|
||||||
|
return undef unless $item;
|
||||||
|
return $item->value;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<get_prop>
|
||||||
|
|
||||||
|
my $prop = $db->get_prop( $key, $prop );
|
||||||
|
|
||||||
|
Shortcut method to get a property from the record defined by the given key.
|
||||||
|
|
||||||
|
Returns undef if the record for that key doesn't exist, or the property does
|
||||||
|
not exist.
|
||||||
|
|
||||||
|
The following code is unsafe if either the key or property doesn't exist:
|
||||||
|
|
||||||
|
my $status = $db->get("foo")->prop('status') || 'disabled';
|
||||||
|
|
||||||
|
and should be written as:
|
||||||
|
|
||||||
|
my $status;
|
||||||
|
|
||||||
|
if (my $s = $db->get("foo"))
|
||||||
|
{
|
||||||
|
$status = $s->prop('status');
|
||||||
|
}
|
||||||
|
|
||||||
|
$status ||= "default";
|
||||||
|
|
||||||
|
With this method, you can use:
|
||||||
|
|
||||||
|
my $value = $db->get_prop("foo", "status") || 'default';
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get_prop
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $item = $self->get(shift);
|
||||||
|
return undef unless $item;
|
||||||
|
return $item->prop(shift);
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<set_value>
|
||||||
|
|
||||||
|
$db->set_value($key, $value)[, create => 1]);
|
||||||
|
|
||||||
|
Shortcut method to set a value to a key in the database without extracting the
|
||||||
|
record first.
|
||||||
|
|
||||||
|
If the record is not pre-existing, it will be created, unless the 'create'
|
||||||
|
option is passed with a value of 0.
|
||||||
|
|
||||||
|
Returns 0 for any errors, 1 for success.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub set_value
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my ($key, $value, %options) = @_;
|
||||||
|
|
||||||
|
my %defaults = (create => 1);
|
||||||
|
%options = (%defaults, %options);
|
||||||
|
|
||||||
|
my $record = $self->get($key);
|
||||||
|
unless ($record)
|
||||||
|
{
|
||||||
|
if ($options{create})
|
||||||
|
{
|
||||||
|
$record = $self->new_record($key, {type => $value})
|
||||||
|
or return 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$record->set_value($value)
|
||||||
|
or return 0;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<set_prop>
|
||||||
|
|
||||||
|
$db->set_prop($key, $prop, $value[, type => $type]);
|
||||||
|
|
||||||
|
Shortcut method to set a property on a record without having to extract the
|
||||||
|
record first.
|
||||||
|
|
||||||
|
If the optional type option is passed, it will be used to create the record if
|
||||||
|
it does not already exist. Otherwise, a non-existent record will cause this
|
||||||
|
method to return an error.
|
||||||
|
|
||||||
|
Returns 0 for any errors, 1 for success.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub set_prop
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my ($key, $prop, $value, %options) = @_;
|
||||||
|
my %defaults = (type => '');
|
||||||
|
%options = (%defaults, %options);
|
||||||
|
|
||||||
|
my $record = $self->get($key);
|
||||||
|
unless ($record)
|
||||||
|
{
|
||||||
|
if ($options{type})
|
||||||
|
{
|
||||||
|
$record = $self->new_record($key, {type => $options{type}})
|
||||||
|
or return 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$record->set_prop($prop, $value)
|
||||||
|
or return 0;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<keys>
|
||||||
|
|
||||||
|
foreach my $key ($db->keys)
|
||||||
|
{
|
||||||
|
|
||||||
|
A simple convenience function to prevent having to access the config hash
|
||||||
|
inside the db object (technically private), or calling map across a get_all
|
||||||
|
call, which is what this is going to do. :)
|
||||||
|
|
||||||
|
This method returns a list of keys to the records in the db. It does not sort.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub keys
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
return map { $_->{key} } $self->get_all;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 migrate
|
||||||
|
|
||||||
|
Process the fragments in the migration queue for this database, using
|
||||||
|
processTemplate.
|
||||||
|
|
||||||
|
The defaults are loaded from /etc/e-smith/db/<dbname>/migrate by default, but
|
||||||
|
the environment variable ESMITH_DB_DEFAULTSDIR can be set to use a different
|
||||||
|
hierarchy if required.
|
||||||
|
|
||||||
|
The entries in "migrate" are perl fragments which will be evaluated and
|
||||||
|
so can munge anything they choose to. But, please be gentle :-)
|
||||||
|
|
||||||
|
So you could have
|
||||||
|
|
||||||
|
/etc/e-smith/db/configuration/migrate/sshd/access
|
||||||
|
|
||||||
|
which is a perl fragment which does something funky to migrate the access
|
||||||
|
property from some old value to some new value.
|
||||||
|
|
||||||
|
After running all the migration scripts, and reloading the DB's data into
|
||||||
|
its local cache, the private method _loadDefaults is called to set any
|
||||||
|
missing default values and any forced settings.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub migrate
|
||||||
|
{
|
||||||
|
my ($self) = @_;
|
||||||
|
|
||||||
|
my $dbfile = basename( $self->{file} );
|
||||||
|
unless ($dbfile)
|
||||||
|
{
|
||||||
|
carp "migrate can't determine filename";
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $defaults_dir = $ENV{ESMITH_DB_DEFAULTSDIR} || "/etc/e-smith/db";
|
||||||
|
my $dir = "$defaults_dir/$dbfile/migrate";
|
||||||
|
|
||||||
|
eval {
|
||||||
|
if ( -d $dir )
|
||||||
|
{
|
||||||
|
processTemplate(
|
||||||
|
{
|
||||||
|
MORE_DATA => { 'DB_FILENAME' => $dbfile },
|
||||||
|
TEMPLATE_PATH => '',
|
||||||
|
OUTPUT_TYPE => 'string',
|
||||||
|
TEMPLATE_EXPAND_QUEUE =>
|
||||||
|
[ $dir, "/etc/e-smith/templates-default" ]
|
||||||
|
}
|
||||||
|
);
|
||||||
|
$self->reload;
|
||||||
|
}
|
||||||
|
$self->_loadDefaults();
|
||||||
|
};
|
||||||
|
if ($@)
|
||||||
|
{
|
||||||
|
warn "Warning: Migration of $dbfile failed fatally: $@\n";
|
||||||
|
$self->set_error($@);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 resetToDefaults
|
||||||
|
|
||||||
|
Reset all entries to their default values, if defaults exist. This calls
|
||||||
|
the internal method _loadDefaults with the forceReset flag set. It should
|
||||||
|
not be used lightly!!
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub resetToDefaults
|
||||||
|
{
|
||||||
|
my ($self) = @_;
|
||||||
|
|
||||||
|
$self->_loadDefaults(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 record_has_defaults
|
||||||
|
|
||||||
|
Returns true if there are defaults or force directories for the
|
||||||
|
given record name
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub record_has_defaults
|
||||||
|
{
|
||||||
|
my ( $self, $name ) = @_;
|
||||||
|
|
||||||
|
my $dbfile = basename( $self->{file} );
|
||||||
|
|
||||||
|
unless ($dbfile)
|
||||||
|
{
|
||||||
|
carp "record_has_defaults can't determine filename";
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
unless ($name)
|
||||||
|
{
|
||||||
|
carp "record_has_defaults can't determine record name";
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $defaults_dir = $ENV{ESMITH_DB_DEFAULTSDIR} || "/etc/e-smith/db";
|
||||||
|
my $dir = "$defaults_dir/$dbfile";
|
||||||
|
|
||||||
|
return ( -d "$dir/defaults/$name" ) || ( -d "$dir/force/$name" );
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 _loadDefaults ($forceReset)
|
||||||
|
|
||||||
|
B<This is a private method.>
|
||||||
|
|
||||||
|
Load the default properties for a given database.
|
||||||
|
Caller can provide a flag to force resetting properties that already exist.
|
||||||
|
|
||||||
|
Any forced properties will be evaluated after setting the default properties.
|
||||||
|
|
||||||
|
The defaults are loaded from the following directories in order (the
|
||||||
|
environment variable ESMITH_DB_DEFAULTSDIR can be set to use a different
|
||||||
|
hierarchy if required):
|
||||||
|
|
||||||
|
/etc/e-smith/db/<dbname>/defaults
|
||||||
|
/etc/e-smith/db/<dbname>/force
|
||||||
|
|
||||||
|
Each of these directories is arranged as a set of subdirectories, with the
|
||||||
|
directory name equal to the key for the given database. With these
|
||||||
|
subdirectories are files, which are named by the properties of these
|
||||||
|
database keys.
|
||||||
|
|
||||||
|
The entries in "defaults" will be skipped if the existing key/property
|
||||||
|
already exists (unless the $forceReset argument is provided). These are
|
||||||
|
simple files, whose contents are the value to be used for that property.
|
||||||
|
|
||||||
|
The entries in "force" are always loaded into the given key/property.
|
||||||
|
These are again simple files, like "defaults".
|
||||||
|
|
||||||
|
To make this concrete, you might have:
|
||||||
|
|
||||||
|
/etc/e-smith/db/configuration/defaults/sshd/access
|
||||||
|
|
||||||
|
containing the single word "private", which would be the default. This
|
||||||
|
value would only be used if no "access" property existed, or the
|
||||||
|
$forceReset option is passed.
|
||||||
|
|
||||||
|
You can override both "defaults" and "migrate" with
|
||||||
|
|
||||||
|
/etc/e-smith/db/configuration/force/sshd/access
|
||||||
|
|
||||||
|
containing the single word "public" to force the value of that property.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _loadDefaults
|
||||||
|
{
|
||||||
|
my ( $self, $force ) = @_;
|
||||||
|
|
||||||
|
my $dbfile = basename( $self->{file} );
|
||||||
|
unless ($dbfile)
|
||||||
|
{
|
||||||
|
carp "_loadDefaults can't determine filename";
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $defaults_dir = $ENV{ESMITH_DB_DEFAULTSDIR} || "/etc/e-smith/db";
|
||||||
|
|
||||||
|
my @propQueue =
|
||||||
|
( "$defaults_dir/$dbfile/defaults", "$defaults_dir/$dbfile/force", );
|
||||||
|
|
||||||
|
foreach my $dir (@propQueue)
|
||||||
|
{
|
||||||
|
|
||||||
|
# Always process the force dir
|
||||||
|
$force = 1 if ( $dir =~ /\/force$/ );
|
||||||
|
|
||||||
|
next unless opendir DH, $dir;
|
||||||
|
foreach my $key ( grep !/^\./, readdir DH )
|
||||||
|
{
|
||||||
|
if ( -d "$dir/$key" )
|
||||||
|
{
|
||||||
|
my %props = ();
|
||||||
|
|
||||||
|
my $rec = $self->get($key);
|
||||||
|
|
||||||
|
opendir DH2, "$dir/$key";
|
||||||
|
foreach my $prop ( grep !/^\./, readdir DH2 )
|
||||||
|
{
|
||||||
|
unless ($force)
|
||||||
|
{
|
||||||
|
next if ( $rec && defined $rec->prop($prop) );
|
||||||
|
}
|
||||||
|
open FH, "$dir/$key/$prop";
|
||||||
|
my $val = join "", (<FH>);
|
||||||
|
chomp $val;
|
||||||
|
|
||||||
|
$props{$prop} = $val;
|
||||||
|
close FH;
|
||||||
|
}
|
||||||
|
closedir DH2;
|
||||||
|
if ($rec)
|
||||||
|
{
|
||||||
|
$rec->merge_props(%props);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$rec = $self->new_record( $key, \%props );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
warn "Found non-directory $key in $dir\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
close DH;
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 get_value_and_delete ($key)
|
||||||
|
|
||||||
|
Retrieve the value of the named key, return it, and delete the record.
|
||||||
|
If the key does not exist, it returns undef. This is normally called from
|
||||||
|
migration code.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get_value_and_delete
|
||||||
|
{
|
||||||
|
my ( $self, $key ) = @_;
|
||||||
|
my $ret;
|
||||||
|
my $rec = $self->get($key);
|
||||||
|
if ($rec)
|
||||||
|
{
|
||||||
|
$ret = $rec->value;
|
||||||
|
$rec->delete;
|
||||||
|
}
|
||||||
|
return $ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 get_prop_and_delete ($key, $prop)
|
||||||
|
|
||||||
|
Retrieve the named property of the named key, return the value, and delete the
|
||||||
|
property from the record. Returns undef if the property or key does not exist.
|
||||||
|
This is normally called from migration code.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get_prop_and_delete
|
||||||
|
{
|
||||||
|
my ( $self, $key, $prop ) = @_;
|
||||||
|
my $ret;
|
||||||
|
my $rec = $self->get($key);
|
||||||
|
if ($rec)
|
||||||
|
{
|
||||||
|
$ret = $rec->prop($prop);
|
||||||
|
$rec->delete_prop($prop);
|
||||||
|
}
|
||||||
|
return $ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub set_error
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
$Error = shift;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub error
|
||||||
|
{
|
||||||
|
return $Error;
|
||||||
|
}
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
SME Server Developers <bugs@e-smith.com>
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
157
root/usr/share/perl5/vendor_perl/esmith/DB/Record.pm
Normal file
157
root/usr/share/perl5/vendor_perl/esmith/DB/Record.pm
Normal file
@ -0,0 +1,157 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::DB::Record;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
use esmith::DB;
|
||||||
|
|
||||||
|
our $VERSION = sprintf '%d.%03d', q$Revision: 1.6 $ =~ /: (\d+).(\d+)/;
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::DB::Record - an individual record in an E-Smith database
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
B<DO NOT USE THIS CLASS DIRECTLY!> use via esmith::DB.
|
||||||
|
|
||||||
|
my $key = $record->key;
|
||||||
|
|
||||||
|
my %properties = $record->props;
|
||||||
|
|
||||||
|
my $value = $record->prop($prop_key);
|
||||||
|
$record->set_prop($prop_key, $prop_val);
|
||||||
|
|
||||||
|
my $value = $record->delete_prop($prop_key);
|
||||||
|
|
||||||
|
$record->merge_props(%more_properties);
|
||||||
|
$record->reset_props(%new_properties);
|
||||||
|
|
||||||
|
$record->delete;
|
||||||
|
|
||||||
|
print $record->show;
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This class is a general interface to individual records in esmith::DB
|
||||||
|
databases. It should not be used directly, but rather esmith::DBs
|
||||||
|
should hand you esmith::DB::Record objects.
|
||||||
|
|
||||||
|
Each subclass of esmith::DB will also have to subclass and implement
|
||||||
|
an esmith::DB::Record subclass.
|
||||||
|
|
||||||
|
|
||||||
|
=head2 Virtual Methods
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item B<key>
|
||||||
|
|
||||||
|
my $key = $record->key;
|
||||||
|
|
||||||
|
Returns the $key for this $record;
|
||||||
|
|
||||||
|
=item B<props>
|
||||||
|
|
||||||
|
my %properties = $record->props;
|
||||||
|
my $num_props = $record->props;
|
||||||
|
|
||||||
|
Returns a hash of all the properties for this $record. In scalar
|
||||||
|
context it will return the number of properties this $record has.
|
||||||
|
|
||||||
|
=item B<prop>
|
||||||
|
|
||||||
|
=item B<set_prop>
|
||||||
|
|
||||||
|
my $value = $record->prop($property);
|
||||||
|
$record->set_prop($property, $value);
|
||||||
|
|
||||||
|
Gets/sets the $value of the $property in this $record.
|
||||||
|
|
||||||
|
set_prop() will die if the database is read-only.
|
||||||
|
|
||||||
|
=item B<delete_prop>
|
||||||
|
|
||||||
|
my $value = $record->delete_prop($property);
|
||||||
|
|
||||||
|
Deletes a $property from the $record, returning the old $value.
|
||||||
|
|
||||||
|
delete_prop() will die if the database is read-only.
|
||||||
|
|
||||||
|
=item B<merge_props>
|
||||||
|
|
||||||
|
$record->merge_props(%properties);
|
||||||
|
|
||||||
|
Adds the %properties to the $records existing properties. Any new
|
||||||
|
keys will be added, any existing keys will be overwritten.
|
||||||
|
|
||||||
|
merge_props() will die if the database is read-only.
|
||||||
|
|
||||||
|
=item B<reset_props>
|
||||||
|
|
||||||
|
$record->reset_props(%properties);
|
||||||
|
|
||||||
|
Replaces the $record's properties with the contents of %properties.
|
||||||
|
Any old properties will be deleted.
|
||||||
|
|
||||||
|
reset_props() will die if the database is read-only.
|
||||||
|
|
||||||
|
=item B<delete>
|
||||||
|
|
||||||
|
$record->delete;
|
||||||
|
|
||||||
|
Deletes the $record from its database.
|
||||||
|
|
||||||
|
delete() will die if the database is read-only.
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
|
||||||
|
=head2 Concrete methods
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item B<show>
|
||||||
|
|
||||||
|
my $formatted = $record->show;
|
||||||
|
|
||||||
|
Returns the $record's key and properties in a nice, human readable
|
||||||
|
format suitable for printing.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub show {
|
||||||
|
my($self) = shift;
|
||||||
|
|
||||||
|
my $out = $self->key."\n";
|
||||||
|
|
||||||
|
my %props = $self->props;
|
||||||
|
|
||||||
|
# Determine our longest key so we know how to format.
|
||||||
|
my $max_len = 0;
|
||||||
|
foreach (keys %props) { $max_len = length if length > $max_len }
|
||||||
|
|
||||||
|
# But don't go too far.
|
||||||
|
$max_len = 40 if $max_len > 40;
|
||||||
|
|
||||||
|
foreach my $prop (sort { $a cmp $b } keys %props) {
|
||||||
|
$out .= sprintf " %${max_len}s = %s\n", $prop, $props{$prop};
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head1 SEE ALSO
|
||||||
|
|
||||||
|
L<esmith::DB>
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
660
root/usr/share/perl5/vendor_perl/esmith/DB/db.pm
Normal file
660
root/usr/share/perl5/vendor_perl/esmith/DB/db.pm
Normal file
@ -0,0 +1,660 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::DB::db;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
use Carp;
|
||||||
|
|
||||||
|
our $VERSION = sprintf '%d.%03d', q$Revision: 1.29 $ =~ /: (\d+).(\d+)/;
|
||||||
|
|
||||||
|
use esmith::db;
|
||||||
|
use esmith::config;
|
||||||
|
use esmith::DB::db::Record;
|
||||||
|
use esmith::DB;
|
||||||
|
our @ISA = qw(esmith::DB);
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
use_ok('esmith::DB::db');
|
||||||
|
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::DB::db - interface to esmith::db databases
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
I<Works just like an esmith::DB class except where noted>
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module provides an abstracted interface to esmith::db flat-file
|
||||||
|
databases. It will read from and write to esmith::db files and can be
|
||||||
|
safely used right along side esmith::db. This follows the esmith::DB
|
||||||
|
interface and will work as documented there unless otherwise stated.
|
||||||
|
|
||||||
|
You should use this instead of esmith::db, and replace any existing
|
||||||
|
esmith::db code with this.
|
||||||
|
|
||||||
|
I<Note for esmith::db users> the old concept of a 'type' is now simply
|
||||||
|
another property.
|
||||||
|
|
||||||
|
my $type = $record->prop('type');
|
||||||
|
|
||||||
|
replaces db_get_type().
|
||||||
|
|
||||||
|
The $record returned by esmith::DB::db subclass is an esmith::DB::db::Record
|
||||||
|
subclass object. See the esmith::DB::db manpage for details on how it is used.
|
||||||
|
|
||||||
|
=head2 Methods
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item B<create>
|
||||||
|
|
||||||
|
Puts its error on esmith::DB::db->error
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$Scratch_Conf = '10e-smith-lib/scratch.conf';
|
||||||
|
unlink $Scratch_Conf;
|
||||||
|
$db = esmith::DB::db->create($Scratch_Conf);
|
||||||
|
END { unlink $Scratch_Conf }
|
||||||
|
|
||||||
|
isa_ok( $db, 'esmith::DB::db', 'create()' );
|
||||||
|
ok( -e $Scratch_Conf, 'created a new config file' );
|
||||||
|
ok(! esmith::DB::db->create($Scratch_Conf),
|
||||||
|
'create() wont walk over an existing config' );
|
||||||
|
like( esmith::DB::db->error, qr/^File exists/, ' right error message' );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub create
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
$file = $class->_file_path($file);
|
||||||
|
my $self;
|
||||||
|
|
||||||
|
eval {
|
||||||
|
$self = $class->_init($file);
|
||||||
|
croak "File exists" if -e $file;
|
||||||
|
|
||||||
|
$self->{config} = $self->_get_config($file)
|
||||||
|
|| croak "Can't get the esmith::config object";
|
||||||
|
|
||||||
|
# touch the config file so it gets created immediately
|
||||||
|
open( FILE, ">>$file" )
|
||||||
|
or die "Failed to open $file for appending: $!\n";
|
||||||
|
close FILE;
|
||||||
|
|
||||||
|
# Migrate, and check for errors, propagating them if they happen.
|
||||||
|
unless ( $self->migrate() )
|
||||||
|
{
|
||||||
|
chomp $@;
|
||||||
|
$self->set_error($@);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
if ($@)
|
||||||
|
{
|
||||||
|
chomp $@;
|
||||||
|
$self->set_error($@);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
return $self;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<open>
|
||||||
|
|
||||||
|
=for notes
|
||||||
|
There's currently no way to get the reason why from esmith::config.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
unlink $Scratch_Conf;
|
||||||
|
ok( !esmith::DB::db->open($Scratch_Conf), 'open() on a non-existent db' );
|
||||||
|
is( esmith::DB::db->error, "File doesn't exist", ' right error' );
|
||||||
|
|
||||||
|
esmith::DB::db->create($Scratch_Conf);
|
||||||
|
$DB = esmith::DB::db->open($Scratch_Conf);
|
||||||
|
isa_ok( $DB, 'esmith::DB::db' );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
$file = $class->_file_path($file);
|
||||||
|
my $self = $class->_init($file);
|
||||||
|
|
||||||
|
if ( -e $file && !-w $file )
|
||||||
|
{
|
||||||
|
$self->{ro} = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $self->_open($file) ? $self : undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<open_local>
|
||||||
|
|
||||||
|
=for notes
|
||||||
|
There's currently no way to get the reason why from esmith::config.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
unlink $Scratch_Conf;
|
||||||
|
ok( !esmith::DB::db->open_local($Scratch_Conf), 'open() on a non-existent db' );
|
||||||
|
is( esmith::DB::db->error, "File doesn't exist", ' right error' );
|
||||||
|
|
||||||
|
esmith::DB::db->create($Scratch_Conf);
|
||||||
|
$DB = esmith::DB::db->open_local($Scratch_Conf);
|
||||||
|
isa_ok( $DB, 'esmith::DB::db' );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open_local
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
$file = $class->_file_path($file);
|
||||||
|
my $self = $class->_init($file);
|
||||||
|
|
||||||
|
if ( -e $file && !-w $file )
|
||||||
|
{
|
||||||
|
$self->{ro} = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $self->_open($file) ? $self : undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
ok( my $db = esmith::DB::db->open_ro($Scratch_Conf),
|
||||||
|
'open_ro on a non-existent db');
|
||||||
|
eval { $db->new_record('foo', { type => 'bar' }) };
|
||||||
|
like( $@, qr/^This DB is opened read-only/ );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open_ro
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
$file = $class->_file_path($file);
|
||||||
|
my $self = $class->_init($file);
|
||||||
|
|
||||||
|
$self->{ro} = 1;
|
||||||
|
|
||||||
|
return $self->_open($file) ? $self : undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
ok( my $db = esmith::DB::db->open_ro_local($Scratch_Conf),
|
||||||
|
'open_ro on a non-existent db');
|
||||||
|
eval { $db->new_record('foo', { type => 'bar' }) };
|
||||||
|
like( $@, qr/^This DB is opened read-only/ );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open_ro_local
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
$file = $class->_file_path($file);
|
||||||
|
my $self = $class->_init($file);
|
||||||
|
|
||||||
|
$self->{ro} = 1;
|
||||||
|
|
||||||
|
return $self->_open($file) ? $self : undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
sub is_ro
|
||||||
|
{
|
||||||
|
return $_[0]->{ro} ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _open
|
||||||
|
{
|
||||||
|
my ( $self, $file ) = @_;
|
||||||
|
|
||||||
|
eval {
|
||||||
|
|
||||||
|
# This is unfortunately not atomic, but I don't think
|
||||||
|
# that's a big deal.
|
||||||
|
die "File doesn't exist\n" unless -e $file;
|
||||||
|
die "File isn't readable\n" unless -r $file;
|
||||||
|
|
||||||
|
$self->{config} = $self->_get_config($file)
|
||||||
|
|| die "Can't get the esmith::config object";
|
||||||
|
};
|
||||||
|
if ($@)
|
||||||
|
{
|
||||||
|
chomp $@;
|
||||||
|
$self->set_error($@);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _get_config
|
||||||
|
{
|
||||||
|
my ( $self, $file ) = @_;
|
||||||
|
|
||||||
|
my %config;
|
||||||
|
tie %config, $self->tie_class, $file;
|
||||||
|
|
||||||
|
return \%config;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _init
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
|
||||||
|
my $self = bless { file => $file }, $class;
|
||||||
|
|
||||||
|
return $self;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _file_path
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
|
||||||
|
if ($file =~ m:/:)
|
||||||
|
{
|
||||||
|
use File::Basename;
|
||||||
|
warn "Deprecated pathname $file passed to _file_path()\n"
|
||||||
|
if dirname($file) eq "/home/e-smith";
|
||||||
|
return $file;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (-e "/home/e-smith/db/$file")
|
||||||
|
{
|
||||||
|
return "/home/e-smith/db/$file";
|
||||||
|
} elsif (-e "/home/e-smith/$file") {
|
||||||
|
warn "Database found in old location /home/e-smith/$file";
|
||||||
|
return "/home/e-smith/$file";
|
||||||
|
} else {
|
||||||
|
return "/home/e-smith/db/$file";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<as_hash>
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use esmith::TestUtils qw(scratch_copy);
|
||||||
|
my $scratch = scratch_copy('10e-smith-lib/db_dummy.conf');
|
||||||
|
my %db = esmith::DB::db->as_hash($scratch);
|
||||||
|
|
||||||
|
my %expect = ( Foo => { type => 'Bar' },
|
||||||
|
Night => { type => 'Day' },
|
||||||
|
Squid => { type => 'cephalopod',
|
||||||
|
arms => 10,
|
||||||
|
species => 'Loligo' },
|
||||||
|
Pipe => { type => 'art',
|
||||||
|
pipe => 'this is not a | got that?',},
|
||||||
|
Haiku => { type => 'poem',
|
||||||
|
words =>
|
||||||
|
"Damian Conway\nGod damn! Damian Conway\nDamian Conway",
|
||||||
|
},
|
||||||
|
Octopus => { type => 'cephalopod',
|
||||||
|
arms => 8,
|
||||||
|
species => '',
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
is_deeply( \%db, \%expect );
|
||||||
|
|
||||||
|
%db = esmith::DB::db->open($scratch)->as_hash;
|
||||||
|
is_deeply( \%db, \%expect );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=item B<reload>
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $db2 = esmith::DB::db->open($Scratch_Conf);
|
||||||
|
my $something = $DB->new_record('something', { type => "wibble" });
|
||||||
|
isa_ok( $something, 'esmith::DB::db::Record', 'new record in 1st DB' );
|
||||||
|
|
||||||
|
ok( !$db2->get('something'), ' 2nd DB still cant see new record' );
|
||||||
|
ok( $db2->reload, ' reload' );
|
||||||
|
ok( $db2->get('something'), ' 2nd DB can see new record' );
|
||||||
|
|
||||||
|
$something->delete;
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub reload
|
||||||
|
{
|
||||||
|
my ($self) = shift;
|
||||||
|
|
||||||
|
$self->_open( $self->file );
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<file>
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
is( $db->file, $Scratch_Conf, 'file()' );
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub file
|
||||||
|
{
|
||||||
|
my ($self) = shift;
|
||||||
|
return $self->{file};
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<new_record>
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $record = $DB->new_record('Big Brother', { year => 1984,
|
||||||
|
day => 'night',
|
||||||
|
type => 'Govt',
|
||||||
|
});
|
||||||
|
isa_ok( $record, 'esmith::DB::db::Record', 'new_record' );
|
||||||
|
is( $record->key, 'Big Brother', 'key' );
|
||||||
|
is( $record->prop('type'), 'Govt', 'type' );
|
||||||
|
is_deeply( {$record->props}, {year => 1984, day => 'night', type => 'Govt'},
|
||||||
|
'props' );
|
||||||
|
is( $record->prop('year'), 1984, 'prop() get' );
|
||||||
|
is( $record->prop('day'), 'night', 'prop() get again' );
|
||||||
|
|
||||||
|
|
||||||
|
$record = $DB->new_record('No props');
|
||||||
|
isa_ok( $record, 'esmith::DB::db::Record', 'new_record() w/o props' );
|
||||||
|
is( $record->key, 'No props', ' key' );
|
||||||
|
|
||||||
|
my $db2 = esmith::DB::db->open($DB->file);
|
||||||
|
ok( $db2->get('No props'), ' can be gotten' );
|
||||||
|
|
||||||
|
$record->delete;
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub new_record
|
||||||
|
{
|
||||||
|
my ( $self, $key, $props ) = @_;
|
||||||
|
|
||||||
|
croak "This DB is opened read-only" if $self->is_ro;
|
||||||
|
|
||||||
|
if ( defined db_get( $self->{config}, $key ) )
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
my $type = exists $props->{type} ? delete $props->{type} : '';
|
||||||
|
db_set( $self->{config}, $key, $type, $props );
|
||||||
|
$self->tie_class->_writeconf($self->{file}, $self->{config});
|
||||||
|
|
||||||
|
return esmith::DB::db::Record->_construct( $self, $key, $self->{config} );
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<get>
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $rec = $DB->get('Big Brother');
|
||||||
|
isa_ok( $rec, 'esmith::DB::db::Record', 'get' );
|
||||||
|
is( $rec->key, 'Big Brother', ' right key' );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get
|
||||||
|
{
|
||||||
|
my ( $self, $key ) = @_;
|
||||||
|
|
||||||
|
unless ( defined db_get( $self->{config}, $key ) )
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return esmith::DB::db::Record->_construct( $self, $key, $self->{config} );
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<get_all>
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$DB->new_record('Borg', { type => 'Govt', resistance => 'futile' });
|
||||||
|
my @records = $DB->get_all;
|
||||||
|
is( @records, 2, 'get_all' );
|
||||||
|
ok( !(grep { !$_->isa('esmith::DB::db::Record') } @records),
|
||||||
|
' theyre all records' );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get_all
|
||||||
|
{
|
||||||
|
my ($self) = shift;
|
||||||
|
|
||||||
|
return
|
||||||
|
map { esmith::DB::db::Record->_construct( $self, $_, $self->{config} ) }
|
||||||
|
db_get( $self->{config} );
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<get_all_by_prop>
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$DB->new_record('Pretz', { type => 'snack', flavor => 'old fashion' });
|
||||||
|
my @records = $DB->get_all_by_prop(type => 'Govt');
|
||||||
|
is( @records, 2, 'get_all_by_prop() type' );
|
||||||
|
ok( !(grep { $_->prop('type') ne 'Govt' } @records),
|
||||||
|
' theyre the right type' );
|
||||||
|
|
||||||
|
$DB->new_record('Pork lips', { type => 'snack', flavor => 'old fashion' });
|
||||||
|
@records = $DB->get_all_by_prop(flavor => 'old fashion');
|
||||||
|
is( @records, 2, 'get_all_by_prop()' );
|
||||||
|
ok( !(grep { $_->prop('flavor') ne 'old fashion' } @records),
|
||||||
|
' they have the right properties' );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub tie_class
|
||||||
|
{
|
||||||
|
return 'esmith::config';
|
||||||
|
}
|
||||||
|
|
||||||
|
sub close
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
=begin deprecated
|
||||||
|
|
||||||
|
=item B<list_by_type>
|
||||||
|
|
||||||
|
Given a type of item to look for in the database (eg "service", "ibay"),
|
||||||
|
returns a list of items which are that type. This is the underlying
|
||||||
|
routine behind esmith::AccountsDB::ibays() and similar methods.
|
||||||
|
|
||||||
|
=end deprecated
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
ok($DB->list_by_type("Govt"), "list_by_type *deprecated*");
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub list_by_type
|
||||||
|
{
|
||||||
|
my ( $self, $type ) = @_;
|
||||||
|
|
||||||
|
return map $_->key, $self->get_all_by_prop( type => $type );
|
||||||
|
}
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head1 EXAMPLE
|
||||||
|
|
||||||
|
The full docs can be found in esmith::DB and esmith::DB::Record, but
|
||||||
|
here's a cheat sheet for esmith::config and esmith::db users.
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item opening the default config
|
||||||
|
|
||||||
|
use esmith::config
|
||||||
|
my %config;
|
||||||
|
tie %config, 'esmith::config;
|
||||||
|
|
||||||
|
Now:
|
||||||
|
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
my $config = esmith::ConfigDB->open;
|
||||||
|
|
||||||
|
=item opening a specific config database
|
||||||
|
|
||||||
|
my %config;
|
||||||
|
tie %config, 'esmith::config', $config_file;
|
||||||
|
|
||||||
|
Now:
|
||||||
|
|
||||||
|
my $config = esmith::ConfigDB->open($config_file);
|
||||||
|
|
||||||
|
=item creating a new config database
|
||||||
|
|
||||||
|
This one's important. Before you could just tie esmith::config to any file
|
||||||
|
and it would create it for you. Now you have to explicitly create it.
|
||||||
|
|
||||||
|
my %config;
|
||||||
|
tie %config, 'esmith::config', $new_config_file;
|
||||||
|
|
||||||
|
Now:
|
||||||
|
|
||||||
|
my $config = esmith::ConfigDB->create($new_config_file);
|
||||||
|
|
||||||
|
=item checking if a record exists
|
||||||
|
|
||||||
|
print "Yep" if exists $config{foo};
|
||||||
|
|
||||||
|
now:
|
||||||
|
|
||||||
|
print "Yep" if $config->get('foo'); # unless of course, 'foo' is zero
|
||||||
|
|
||||||
|
=item creating a new record
|
||||||
|
|
||||||
|
Previously you could just create records on the fly:
|
||||||
|
|
||||||
|
# single value
|
||||||
|
$config{foo} = 'whatever';
|
||||||
|
|
||||||
|
# with properties
|
||||||
|
db_set(\%config, 'whatever', 'sometype', { with => 'properties' });
|
||||||
|
|
||||||
|
Now you have to explicitly create them:
|
||||||
|
|
||||||
|
# single value
|
||||||
|
my $foo = $config->new_record('foo');
|
||||||
|
$foo->set_value('foo');
|
||||||
|
|
||||||
|
# with properties
|
||||||
|
my %defaults = ( 'type' => 'sometype',
|
||||||
|
'linux' => 'stable',
|
||||||
|
'windows' => 'stable?' );
|
||||||
|
my $foo = $config->new_record('foo', \%defaults);
|
||||||
|
|
||||||
|
Note that 'type' is now just another property.
|
||||||
|
|
||||||
|
Here's a handy "create this if it doesn't already exist" idiom.
|
||||||
|
|
||||||
|
my $rec = $config->get($key) ||
|
||||||
|
$config->new_record($key);
|
||||||
|
|
||||||
|
=item getting a value
|
||||||
|
|
||||||
|
Entries in a database should no longer be thought of as values, but as
|
||||||
|
records.
|
||||||
|
|
||||||
|
my $val = $config{foo};
|
||||||
|
|
||||||
|
Now this only works with entries with single value. Things with
|
||||||
|
multiple properties are dealt with differently.
|
||||||
|
|
||||||
|
my $record = $config->get('foo');
|
||||||
|
my $val = $record->value;
|
||||||
|
|
||||||
|
=item setting a value
|
||||||
|
|
||||||
|
$config{foo} = 'something';
|
||||||
|
|
||||||
|
now
|
||||||
|
|
||||||
|
my $record = $config->get('foo');
|
||||||
|
$record->set_value('something');
|
||||||
|
|
||||||
|
=item getting a property
|
||||||
|
|
||||||
|
my $this = db_get_prop(\%config, 'foo', 'this');
|
||||||
|
|
||||||
|
now:
|
||||||
|
|
||||||
|
my $foo = $config->get('foo');
|
||||||
|
my $this = $foo->prop('this');
|
||||||
|
|
||||||
|
=item getting & setting properties
|
||||||
|
|
||||||
|
my $val = db_get_prop(\%config, 'foo', 'some prop');
|
||||||
|
db_set_prop(\%config, 'foo', 'some prop' => $new_val);
|
||||||
|
|
||||||
|
now:
|
||||||
|
|
||||||
|
my $val = $record->prop('some prop');
|
||||||
|
$record->set_prop('some prop' => $new_val);
|
||||||
|
|
||||||
|
=item get/setting the type
|
||||||
|
|
||||||
|
my $type = db_get_type(\%config, 'foo');
|
||||||
|
db_set_type(\%config, 'foo', $new_type);
|
||||||
|
|
||||||
|
type is now just a property
|
||||||
|
|
||||||
|
my $record = $db->get('foo');
|
||||||
|
my $type = $record->prop('type');
|
||||||
|
$record->set_prop('type', $new_type);
|
||||||
|
|
||||||
|
=item getting all the properties
|
||||||
|
|
||||||
|
my %props = db_get_prop(\%config, 'foo');
|
||||||
|
|
||||||
|
now
|
||||||
|
|
||||||
|
my %props = $record->props;
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
SME Server Developers <bugs@e-smith.com>
|
||||||
|
|
||||||
|
=head1 SEE ALSO
|
||||||
|
|
||||||
|
L<esmith::AccountsDB>, L<esmith::ConfigDB>, L<esmith::DB::db::Record>
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
353
root/usr/share/perl5/vendor_perl/esmith/DB/db/Record.pm
Normal file
353
root/usr/share/perl5/vendor_perl/esmith/DB/db/Record.pm
Normal file
@ -0,0 +1,353 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::DB::db::Record;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
use Carp;
|
||||||
|
|
||||||
|
use esmith::db;
|
||||||
|
use esmith::DB::db;
|
||||||
|
|
||||||
|
require esmith::DB::Record;
|
||||||
|
our @ISA = qw(esmith::DB::Record);
|
||||||
|
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use_ok('esmith::DB::db::Record');
|
||||||
|
use_ok('esmith::DB::db');
|
||||||
|
|
||||||
|
use File::Copy;
|
||||||
|
$Scratch_Conf = '10e-smith-lib/scratch.conf';
|
||||||
|
copy('10e-smith-lib/db_dummy.conf', $Scratch_Conf);
|
||||||
|
END { unlink $Scratch_Conf }
|
||||||
|
|
||||||
|
$DB = esmith::DB::db->open($Scratch_Conf);
|
||||||
|
$Squid = $DB->get('Squid');
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::DB::db::Record - Individual records in an esmith::db database
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
Unless otherwise noted, works just like esmith::DB::Record.
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This class represents entries in esmith::db flat-file database. A
|
||||||
|
single object is a single line.
|
||||||
|
|
||||||
|
This class is not useful by itself but rather they are handed out
|
||||||
|
via esmith::DB::db objects.
|
||||||
|
|
||||||
|
|
||||||
|
=begin protected
|
||||||
|
|
||||||
|
=head2 Protected Methods
|
||||||
|
|
||||||
|
These methods are only allowed to be called by esmith::DB::db classes.
|
||||||
|
|
||||||
|
=item B<_construct>
|
||||||
|
|
||||||
|
my $record = esmith::DB::db::Record->_construct($db, $key, $config);
|
||||||
|
|
||||||
|
Generates a new esmith::DB::db::Record representing data inside the
|
||||||
|
$db (an esmith::DB::db object).
|
||||||
|
|
||||||
|
This does *not* write anything into $db. This is here so a $db can
|
||||||
|
initialize a new Record from existing data.
|
||||||
|
|
||||||
|
=end protected
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _construct {
|
||||||
|
my($class, $db, $key, $config) = @_;
|
||||||
|
|
||||||
|
die "_construct may only be called by esmith::DB::db"
|
||||||
|
unless caller->isa('esmith::DB::db');
|
||||||
|
|
||||||
|
my $self = {
|
||||||
|
db => $db,
|
||||||
|
config => $config,
|
||||||
|
key => $key
|
||||||
|
};
|
||||||
|
|
||||||
|
return bless $self, $class;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 Methods
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item B<key>
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
is( $Squid->key, 'Squid', 'key()' );
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub key {
|
||||||
|
my($self) = shift;
|
||||||
|
return $self->{key};
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<props>
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
is_deeply( {$Squid->props}, {arms => 10, species => 'Loligo',
|
||||||
|
type => 'cephalopod'}, 'props()' );
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub props {
|
||||||
|
my($self) = shift;
|
||||||
|
|
||||||
|
my %props = db_get_prop($self->{config}, $self->{key});
|
||||||
|
$props{type} = db_get_type($self->{config}, $self->{key});
|
||||||
|
foreach my $prop (keys %props) {
|
||||||
|
$props{$prop} =~ s{\\\|}{\|}g if $props{$prop};
|
||||||
|
}
|
||||||
|
return wantarray ? %props : keys %props;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<prop>
|
||||||
|
|
||||||
|
=item B<set_prop>
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
is( $Squid->prop('arms'), 10, 'prop()' );
|
||||||
|
$Squid->set_prop('arms', 1000);
|
||||||
|
is( $Squid->prop('arms'), 1000, 'set_prop()' );
|
||||||
|
|
||||||
|
is( $Squid->prop('type'), 'cephalopod', 'prop() type get' );
|
||||||
|
$Squid->set_prop('type', 'tree dweller');
|
||||||
|
is( $Squid->prop('type'), 'tree dweller', 'set_prop() type set' );
|
||||||
|
|
||||||
|
$Squid->set_prop('bar', 'foo | bar');
|
||||||
|
is( $Squid->prop('bar'), 'foo bar', 'prop/set_prop with pipes - pipe stripped' );
|
||||||
|
|
||||||
|
{
|
||||||
|
my $warning = '';
|
||||||
|
local $SIG{__WARN__} = sub { $warning = join '', @_ };
|
||||||
|
$Squid->prop('bar', 'foo');
|
||||||
|
like( $warning, qr/^prop\(\) got extra arguments 'foo'. Maybe you ment set_prop\(\)\?/, 'prop()/set_prop() mixup warns' );
|
||||||
|
|
||||||
|
$warning = '';
|
||||||
|
is( $Squid->prop('I_dont_exist'), undef, 'prop() on non-existent prop' );
|
||||||
|
is( $warning, '', ' no warning' );
|
||||||
|
|
||||||
|
$warning = '';
|
||||||
|
$Squid->set_prop('I_dont_exist', undef);
|
||||||
|
is( $Squid->prop('I_dont_exist'), '', 'set_prop() with undef value' );
|
||||||
|
is( $warning, '', ' no warning' );
|
||||||
|
$Squid->delete_prop('I_dont_exist');
|
||||||
|
}
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub prop {
|
||||||
|
my($self, $property) = splice @_, 0, 2;
|
||||||
|
|
||||||
|
warn sprintf "prop() got extra arguments '%s'. Maybe you ment set_prop()?",
|
||||||
|
"@_" if @_;
|
||||||
|
|
||||||
|
my $value;
|
||||||
|
if( $property eq 'type' ) {
|
||||||
|
$value = db_get_type($self->{config}, $self->{key});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$value = db_get_prop($self->{config}, $self->{key}, $property);
|
||||||
|
}
|
||||||
|
|
||||||
|
# Unescape escaped pipes. esmith::db can't do this for us.
|
||||||
|
$value =~ s{\\\|}{\|}g if defined $value;
|
||||||
|
|
||||||
|
return $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub set_prop {
|
||||||
|
my($self, $property, $value) = @_;
|
||||||
|
|
||||||
|
croak "The DB is open read-only" if $self->{db}->is_ro;
|
||||||
|
|
||||||
|
# Strip pipes - we can't safely escape them while some code
|
||||||
|
# still expects to split on pipe
|
||||||
|
$value =~ s{\|}{}g if defined $value;
|
||||||
|
|
||||||
|
my $ret;
|
||||||
|
if( $property eq 'type' ) {
|
||||||
|
$ret = db_set_type($self->{config}, $self->{key}, $value);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$ret = db_set_prop($self->{config}, $self->{key},
|
||||||
|
$property => $value);
|
||||||
|
}
|
||||||
|
return $ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<delete_prop>
|
||||||
|
|
||||||
|
A special case for esmith::DB::db::Record, you're not allowed to
|
||||||
|
delete the 'type' property.
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
is( $Squid->delete_prop('species'), 'Loligo',
|
||||||
|
'delete_prop() returns the old value' );
|
||||||
|
is_deeply( {$Squid->props}, {arms => 1000, bar => 'foo bar',
|
||||||
|
type => 'tree dweller' },
|
||||||
|
' and deletes' );
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub delete_prop {
|
||||||
|
my($self, $property) = @_;
|
||||||
|
|
||||||
|
croak "The DB is open read-only" if $self->{db}->is_ro;
|
||||||
|
|
||||||
|
croak "You're not allowed to delete a type from an esmith::DB::db::Record"
|
||||||
|
if $property eq 'type';
|
||||||
|
|
||||||
|
my $val = $self->prop($property);
|
||||||
|
db_delete_prop($self->{config}, $self->{key}, $property);
|
||||||
|
|
||||||
|
return $val;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<merge_props>
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $octopus = $DB->get('Octopus');
|
||||||
|
$octopus->merge_props( arms => '8 + 2i', name => 'Fluffy', pipe => 'not |');
|
||||||
|
is_deeply( {$octopus->props}, { arms => '8 + 2i', type => 'cephalopod',
|
||||||
|
species => '', name => 'Fluffy',
|
||||||
|
pipe => 'not ' }, 'merge_props()' );
|
||||||
|
$octopus->merge_props( type => 'foo' );
|
||||||
|
is_deeply( {$octopus->props}, { arms => '8 + 2i', type => 'foo',
|
||||||
|
species => '', name => 'Fluffy',
|
||||||
|
pipe => 'not ' }, ' with type' );
|
||||||
|
|
||||||
|
$octopus->merge_props( { type => 'foo' } );
|
||||||
|
like( $_STDERR_, qr/^merge_props\(\) was accidentally passed a hash ref/m,
|
||||||
|
' anti-hash ref protection');
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub merge_props {
|
||||||
|
my($self, %new_props) = @_;
|
||||||
|
|
||||||
|
croak "The DB is open read-only" if $self->{db}->is_ro;
|
||||||
|
|
||||||
|
if( ref $_[1] ) {
|
||||||
|
carp("merge_props() was accidentally passed a hash ref");
|
||||||
|
}
|
||||||
|
|
||||||
|
my %props = $self->props;
|
||||||
|
my %merged_props = (%props, %new_props);
|
||||||
|
|
||||||
|
# Strip out pipes.
|
||||||
|
foreach my $prop (keys %merged_props) {
|
||||||
|
$merged_props{$prop} =~ s{\|}{}g
|
||||||
|
if defined $merged_props{$prop};
|
||||||
|
}
|
||||||
|
|
||||||
|
my $type = delete $merged_props{type};
|
||||||
|
db_set($self->{config}, $self->{key}, $type, \%merged_props);
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<reset_props>
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $octopus = $DB->get('Octopus');
|
||||||
|
|
||||||
|
eval { $octopus->reset_props( { type => 'foo' } ); };
|
||||||
|
like( $_STDERR_, qr/^reset_props\(\) was accidentally passed a hash ref/m,
|
||||||
|
' anti-hash ref protection');
|
||||||
|
|
||||||
|
$octopus->reset_props( arms => 8, name => 'Rupert', type => 'foo' );
|
||||||
|
is_deeply( {$octopus->props}, { arms => '8', name => 'Rupert',
|
||||||
|
type => 'foo' }, 'reset_props' );
|
||||||
|
|
||||||
|
eval { $octopus->reset_props( arms => '8 + 2i', name => 'Fluffy',
|
||||||
|
pipe => 'not ') };
|
||||||
|
like( $@, qr/^You must have a type property/, ' you must have a type');
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub reset_props {
|
||||||
|
my($self, %new_props) = @_;
|
||||||
|
|
||||||
|
croak "The DB is open read-only" if $self->{db}->is_ro;
|
||||||
|
|
||||||
|
if( ref $_[1] ) {
|
||||||
|
carp("reset_props() was accidentally passed a hash ref");
|
||||||
|
}
|
||||||
|
|
||||||
|
die "You must have a type property" unless $new_props{type};
|
||||||
|
|
||||||
|
# Strip out pipes
|
||||||
|
foreach my $prop (keys %new_props) {
|
||||||
|
$new_props{$prop} =~ s{\|}{}g
|
||||||
|
if defined $new_props{$prop};
|
||||||
|
}
|
||||||
|
|
||||||
|
my $type = delete $new_props{type} || $self->prop('type');
|
||||||
|
db_set($self->{config}, $self->{key}, $type, \%new_props);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<delete>
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
my $foo = $DB->get('Foo');
|
||||||
|
$foo->delete;
|
||||||
|
ok( !$DB->get('Foo'), 'delete()' );
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub delete {
|
||||||
|
my($self) = shift;
|
||||||
|
|
||||||
|
croak "The DB is open read-only" if $self->{db}->is_ro;
|
||||||
|
db_delete($self->{config}, $self->{key});
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<show>
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
is( $Squid->show, <<SQUID, 'show' );
|
||||||
|
Squid
|
||||||
|
arms = 1000
|
||||||
|
bar = foo bar
|
||||||
|
type = tree dweller
|
||||||
|
SQUID
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head1 SEE ALSO
|
||||||
|
|
||||||
|
L<esmith::DB::db>
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
99
root/usr/share/perl5/vendor_perl/esmith/DomainsDB.pm
Normal file
99
root/usr/share/perl5/vendor_perl/esmith/DomainsDB.pm
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::DomainsDB;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
|
||||||
|
use esmith::DB::db;
|
||||||
|
our @ISA = qw( esmith::DB::db );
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::DomainsDB - interface to esmith domains database
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::DomainsDB;
|
||||||
|
my $c = esmith::DomainsDB->open;
|
||||||
|
|
||||||
|
# everything else works just like esmith::DB::db
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module provides an abstracted interface to the esmith domain
|
||||||
|
database.
|
||||||
|
|
||||||
|
Unless otherwise noted, esmith::DomainsDB acts like esmith::DB::db.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
=head2 open()
|
||||||
|
|
||||||
|
Like esmith::DB->open, but if given no $file it will try to open the
|
||||||
|
file in the ESMITH_DOMAINS_DB environment variable or domains.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use_ok("esmith::DomainsDB");
|
||||||
|
|
||||||
|
$C = esmith::DomainsDB->open('10e-smith-lib/domains.conf');
|
||||||
|
isa_ok($C, 'esmith::DomainsDB');
|
||||||
|
is( $C->get("test")->prop('foo'), "bar",
|
||||||
|
"We can get stuff from the db");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open {
|
||||||
|
my ($self, $file) = @_;
|
||||||
|
$file = $file || $ENV{ESMITH_DOMAINS_DB} || "domains";
|
||||||
|
return $self->SUPER::open($file);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 open_ro()
|
||||||
|
|
||||||
|
Like esmith::DB->open_ro, but if given no $file it will try to open the
|
||||||
|
file in the ESMITH_DOMAINS_DB environment variable or domains.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open_ro {
|
||||||
|
my($class, $file) = @_;
|
||||||
|
$file = $file || $ENV{ESMITH_DOMAINS_DB} || "domains";
|
||||||
|
return $class->SUPER::open_ro($file);
|
||||||
|
}
|
||||||
|
=for testing
|
||||||
|
$C = esmith::DomainsDB->open('10e-smith-lib/domains.conf');
|
||||||
|
isa_ok($C, 'esmith::DomainsDB');
|
||||||
|
can_ok($C, 'domains');
|
||||||
|
can_ok($C, 'get_all_by_prop');
|
||||||
|
is(scalar($C->domains()), 2, "Found 2 domains with domains()");
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub domains {
|
||||||
|
my ($self) = @_;
|
||||||
|
return $self->get_all_by_prop(type => 'domain');
|
||||||
|
}
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
SME Server Developers <bugs@e-smith.com>
|
||||||
|
|
||||||
|
=head1 SEE ALSO
|
||||||
|
|
||||||
|
L<esmith::ConfigDB>
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
230
root/usr/share/perl5/vendor_perl/esmith/HostsDB.pm
Normal file
230
root/usr/share/perl5/vendor_perl/esmith/HostsDB.pm
Normal file
@ -0,0 +1,230 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::HostsDB;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
|
||||||
|
use esmith::DB::db;
|
||||||
|
our @ISA = qw( esmith::DB::db );
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::HostsDB - interface to esmith hostnames/addresses database
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::HostsDB;
|
||||||
|
my $hosts = esmith::HostsDB->open;
|
||||||
|
|
||||||
|
# everything else works just like esmith::DB::db
|
||||||
|
|
||||||
|
# these methods are added
|
||||||
|
my @hosts = $hosts->hosts;
|
||||||
|
my @new_hosts = $hosts->propogate_hosts;
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module provides an abstracted interface to the esmith hosts
|
||||||
|
database.
|
||||||
|
|
||||||
|
Unless otherwise noted, esmith::HostsDB acts like esmith::DB::db.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
=head2 Overridden methods
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item I<open>
|
||||||
|
|
||||||
|
Like esmith::DB->open, but if given no $file it will try to open the
|
||||||
|
file in the ESMITH_HOSTS_DB environment variable or hosts.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use_ok("esmith::HostsDB");
|
||||||
|
|
||||||
|
$H = esmith::HostsDB->open('10e-smith-lib/hosts.conf');
|
||||||
|
isa_ok($H, 'esmith::HostsDB');
|
||||||
|
is( $H->get("otherhost.mydomain.xxx")->prop('InternalIP'), "192.168.1.3",
|
||||||
|
"We can get stuff from the db");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open {
|
||||||
|
my($class, $file) = @_;
|
||||||
|
$file = $file || $ENV{ESMITH_HOSTS_DB} || "hosts";
|
||||||
|
return $class->SUPER::open($file);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 open_ro()
|
||||||
|
|
||||||
|
Like esmith::DB->open_ro, but if given no $file it will try to open the
|
||||||
|
file in the ESMITH_HOSTS_DB environment variable or hosts.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open_ro {
|
||||||
|
my($class, $file) = @_;
|
||||||
|
$file = $file || $ENV{ESMITH_HOSTS_DB} || "hosts";
|
||||||
|
return $class->SUPER::open_ro($file);
|
||||||
|
}
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head2 Additional Methods
|
||||||
|
|
||||||
|
These methods are added be esmith::HostsDB
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item I<hosts>
|
||||||
|
|
||||||
|
my @hosts = $hosts->hosts;
|
||||||
|
|
||||||
|
Returns a list of all host records in the database.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$db = esmith::HostsDB->open('10e-smith-lib/hosts.conf');
|
||||||
|
isa_ok($db, 'esmith::HostsDB');
|
||||||
|
can_ok($db, 'hosts');
|
||||||
|
my @hosts = $db->hosts();
|
||||||
|
isnt( @hosts, 0 );
|
||||||
|
is_deeply(\@hosts, [$db->get_all_by_prop('type' => 'host')]);
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub hosts {
|
||||||
|
my ($self) = @_;
|
||||||
|
return $self->get_all_by_prop('type' => 'host');
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<propogate_hosts>
|
||||||
|
|
||||||
|
my @new_hosts = $hosts->propogate_hosts($old_name, $new_name);
|
||||||
|
|
||||||
|
When the name of your e-smith machine changes, this will change the
|
||||||
|
name of any hosts which also started with $old_name to use the
|
||||||
|
$new_name.
|
||||||
|
|
||||||
|
Returns a list of the newly tranlsated host records.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
|
||||||
|
my $hosts_file = '10e-smith-lib/propogate_hosts.conf';
|
||||||
|
END { unlink $hosts_file }
|
||||||
|
|
||||||
|
my $db = esmith::HostsDB->create($hosts_file);
|
||||||
|
|
||||||
|
use esmith::TestUtils qw(scratch_copy);
|
||||||
|
my $c_scratch = scratch_copy('10e-smith-lib/configuration.conf');
|
||||||
|
my $config = esmith::ConfigDB->open($c_scratch);
|
||||||
|
isa_ok($config, 'esmith::ConfigDB');
|
||||||
|
|
||||||
|
my $name = $config->get('SystemName')->value;
|
||||||
|
|
||||||
|
# setup some dummy hosts to propogate.
|
||||||
|
foreach my $host ( "$name.tofu-dog.com", "$name.wibble.org",
|
||||||
|
"wibble.$name.org", "yarrow.hack" )
|
||||||
|
{
|
||||||
|
$db->new_record($host, { type => 'host', HostType => 'Self',
|
||||||
|
ExternalIP => '', InternalIP => ''
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
$db->reload;
|
||||||
|
my @new_hosts = $db->propogate_hosts($name, "armondo");
|
||||||
|
my @hosts = $db->hosts;
|
||||||
|
is( @hosts, 4 );
|
||||||
|
is_deeply( [sort map { $_->key } @hosts],
|
||||||
|
[sort +('armondo.tofu-dog.com',
|
||||||
|
'armondo.wibble.org',
|
||||||
|
"wibble.$name.org",
|
||||||
|
'yarrow.hack',
|
||||||
|
)]
|
||||||
|
);
|
||||||
|
|
||||||
|
is( @new_hosts, 2 );
|
||||||
|
is_deeply( [sort map { $_->key } @new_hosts],
|
||||||
|
[sort qw(armondo.tofu-dog.com armondo.wibble.org)]
|
||||||
|
);
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub propogate_hosts
|
||||||
|
{
|
||||||
|
my($self, $old_name, $new_name) = @_;
|
||||||
|
|
||||||
|
my @new_hosts = ();
|
||||||
|
foreach my $host ($self->hosts)
|
||||||
|
{
|
||||||
|
my $new_host = $host->key;
|
||||||
|
if( $new_host =~ s/^\Q$old_name.\E/$new_name./ )
|
||||||
|
{
|
||||||
|
push @new_hosts, $self->new_record($new_host,
|
||||||
|
{ $host->props }
|
||||||
|
);
|
||||||
|
$host->delete;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return @new_hosts;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=head2 $db->get_hosts_by_domain
|
||||||
|
|
||||||
|
Given a domain name (as a string), finds any hosts which match it and
|
||||||
|
return them as a list of record objects.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $h = esmith::HostsDB->open('10e-smith-lib/hosts.conf');
|
||||||
|
my @hosts = $h->get_hosts_by_domain('otherdomain.xxx');
|
||||||
|
is(scalar(@hosts), 2, "Found two hosts in otherdomain.xxx");
|
||||||
|
isa_ok($hosts[0], 'esmith::DB::Record');
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get_hosts_by_domain {
|
||||||
|
my ($self, $domain) = @_;
|
||||||
|
my @all = $self->get_all();
|
||||||
|
my @return;
|
||||||
|
foreach my $h (@all) {
|
||||||
|
push @return, $h if $h->key() =~ /^[^\.]+\.$domain$/;
|
||||||
|
}
|
||||||
|
return @return;
|
||||||
|
}
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
SME Server Developers <bugs@e-smith.com>
|
||||||
|
|
||||||
|
=head1 SEE ALSO
|
||||||
|
|
||||||
|
L<esmith::ConfigDB>
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
331
root/usr/share/perl5/vendor_perl/esmith/I18N.pm
Normal file
331
root/usr/share/perl5/vendor_perl/esmith/I18N.pm
Normal file
@ -0,0 +1,331 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::I18N;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
use POSIX qw(setlocale LC_ALL LC_CTYPE);
|
||||||
|
use Locale::gettext;
|
||||||
|
use I18N::AcceptLanguage;
|
||||||
|
use I18N::LangTags qw(is_language_tag locale2language_tag);
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::I18N - Internationalization utilities Mitel Network SME Server
|
||||||
|
|
||||||
|
=head1 VERSION
|
||||||
|
|
||||||
|
This file documents C<esmith::I18N> version B<1.4.0>
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::I18N;
|
||||||
|
|
||||||
|
my $i18n = new esmith::I18N;
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module provides general internationalization and localisation
|
||||||
|
utilities for developers of the Mitel Networks SME Server.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use I18N::LangTags qw(is_language_tag locale2language_tag language_tag2locale);
|
||||||
|
use_ok('esmith::I18N');
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=head1 GENERAL UTILITIES
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub new
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $class = ref($self) || $self;
|
||||||
|
|
||||||
|
my %args = @_;
|
||||||
|
|
||||||
|
return $self;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 availableLocales()
|
||||||
|
|
||||||
|
Returns an array containing the available locales supported by the
|
||||||
|
server.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$ENV{ESMITH_CONFIG_DB}="10e-smith-lib/sysconfig-en_US.conf";
|
||||||
|
$ENV{ESMITH_I18N_USRSHARELOCALE}="10e-smith-lib/usr/share/locale";
|
||||||
|
|
||||||
|
my $i18n = new esmith::I18N;
|
||||||
|
my @locales = grep !/CVS/, sort $i18n->availableLocales;
|
||||||
|
|
||||||
|
# NOTE: de is not a valid locale for the test - no server-console file
|
||||||
|
is_deeply(\@locales, [('en_US', 'fr_CA', 'wx_YZ')], "Locales match" );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub availableLocales()
|
||||||
|
{
|
||||||
|
my ($self) = shift;
|
||||||
|
|
||||||
|
my $localedir = $ENV{ESMITH_I18N_USRSHARELOCALE} || '/usr/share/locale';
|
||||||
|
|
||||||
|
return () unless opendir LOCALE, $localedir;
|
||||||
|
|
||||||
|
my @locales;
|
||||||
|
|
||||||
|
foreach my $locale ( grep(!/\./, readdir LOCALE) )
|
||||||
|
{
|
||||||
|
push @locales, $locale if
|
||||||
|
(-f "$localedir/$locale/LC_MESSAGES/server-console.mo" or
|
||||||
|
-f "$localedir/$locale/LC_MESSAGES/server-console.po");
|
||||||
|
}
|
||||||
|
|
||||||
|
closedir LOCALE;
|
||||||
|
return @locales;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 fallbackLocale()
|
||||||
|
|
||||||
|
Return system fallback locale
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub fallbackLocale()
|
||||||
|
{
|
||||||
|
return "en_US.utf8";
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 preferredLocale()
|
||||||
|
|
||||||
|
Retrieves the preferred locale for this server.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$ENV{ESMITH_CONFIG_DB}="10e-smith-lib/sysconfig-en_US.conf";
|
||||||
|
|
||||||
|
my $i18n = new esmith::I18N;
|
||||||
|
is($i18n->preferredLocale, 'en_US', "en_US.conf: Preferred locale is en_US");
|
||||||
|
|
||||||
|
$ENV{ESMITH_CONFIG_DB}="10e-smith-lib/sysconfig-fr_CA.conf";
|
||||||
|
|
||||||
|
$i18n = new esmith::I18N;
|
||||||
|
is($i18n->preferredLocale, 'fr_CA', "fr_CA.conf: Preferred locale is fr_CA");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub preferredLocale()
|
||||||
|
{
|
||||||
|
my ($self) = shift;
|
||||||
|
|
||||||
|
my $db = esmith::ConfigDB->open_ro || return $self->fallbackLocale;
|
||||||
|
|
||||||
|
my ($locale, @rest) = $db->getLocale();
|
||||||
|
|
||||||
|
return $locale || $self->fallbackLocale;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 setLocale()
|
||||||
|
|
||||||
|
Configure the locale for gettext() for the supplied text domain.
|
||||||
|
|
||||||
|
The method takes two arguments, the text domain, and an optional argument
|
||||||
|
which can be either a language tag or a locale.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub setLocale()
|
||||||
|
{
|
||||||
|
my ($self, $text_domain, $opt) = @_;
|
||||||
|
my $locale;
|
||||||
|
|
||||||
|
$locale = $self->langtag2locale($opt) if ($opt);
|
||||||
|
$locale ||= $self->preferredLocale;
|
||||||
|
$locale =~ s{(?:\..*)?$}{.utf8};
|
||||||
|
|
||||||
|
$ENV{'LANGUAGE'} = $locale;
|
||||||
|
$ENV{'LANG'} = $ENV{'LANGUAGE'};
|
||||||
|
|
||||||
|
setlocale(LC_MESSAGES, $locale);
|
||||||
|
setlocale(LC_MESSAGES, $locale);
|
||||||
|
setlocale(LC_ALL, $locale);
|
||||||
|
setlocale(LC_ALL, $locale);
|
||||||
|
|
||||||
|
bindtextdomain ($text_domain, "/usr/share/locale");
|
||||||
|
textdomain ($text_domain);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=head2 langtag2locale
|
||||||
|
|
||||||
|
Even though the directories appear in /usr/share/locale, they also need
|
||||||
|
to appear in /usr/lib/locale to actually be treated as locales. Read the
|
||||||
|
Perl locale docs for details of how horrid this is. For now, we're just
|
||||||
|
going to force things for supported languages.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
my $i18n = new esmith::I18N;
|
||||||
|
|
||||||
|
is($i18n->langtag2locale("en"), "en_US", "en langtag is en_US locale");
|
||||||
|
is($i18n->langtag2locale("en-us"), "en_US", "en-us langtag is en_US locale");
|
||||||
|
is($i18n->langtag2locale("en-au"), "en_AU", "en-au langtag is en_AU locale");
|
||||||
|
|
||||||
|
is($i18n->langtag2locale("es"), "es_ES", "es langtag is es_ES locale");
|
||||||
|
is($i18n->langtag2locale("es-es"), "es_ES", "es-es langtag is es_ES locale");
|
||||||
|
is($i18n->langtag2locale("es-ar"), "es_AR", "es-ar langtag is es_AR locale");
|
||||||
|
|
||||||
|
is($i18n->langtag2locale("fr"), "fr_CA", "fr langtag is fr_CA locale");
|
||||||
|
is($i18n->langtag2locale("fr-ca"), "fr_CA", "fr-ca langtag is fr_CA locale");
|
||||||
|
is($i18n->langtag2locale("fr-fr"), "fr_FR", "fr-fr langtag is fr_FR locale");
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub langtag2locale
|
||||||
|
{
|
||||||
|
my ($self, $opt) = @_;
|
||||||
|
|
||||||
|
my $locale;
|
||||||
|
|
||||||
|
if (is_language_tag($opt))
|
||||||
|
{
|
||||||
|
$locale = _language_tag2locale($opt) || $self->fallbackLocale;
|
||||||
|
|
||||||
|
unless (-d "/usr/lib/locale/$locale")
|
||||||
|
{
|
||||||
|
$locale = "da_DK" if ($opt =~ /^da(-.*)?/);
|
||||||
|
$locale = "de_DE" if ($opt =~ /^de(-.*)?/);
|
||||||
|
$locale = "el_GR" if ($opt =~ /^el(-.*)?/);
|
||||||
|
$locale = "en_US" if ($opt =~ /^en(-.*)?/);
|
||||||
|
$locale = "es_ES" if ($opt =~ /^es(-.*)?/);
|
||||||
|
$locale = "fr_CA" if ($opt =~ /^fr(-.*)?/);
|
||||||
|
$locale = "hu_HU" if ($opt =~ /^hu(-.*)?/);
|
||||||
|
$locale = "id_ID" if ($opt =~ /^id(-.*)?/);
|
||||||
|
$locale = "it_IT" if ($opt =~ /^it(-.*)?/);
|
||||||
|
$locale = "nl_NL" if ($opt =~ /^nl(-.*)?/);
|
||||||
|
$locale = "pt_BR" if ($opt =~ /^pt(-.*)?/);
|
||||||
|
$locale = "sl_SL" if ($opt =~ /^sl(-.*)?/);
|
||||||
|
$locale = "sv_SE" if ($opt =~ /^sv(-.*)?/);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$locale = $opt;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $locale;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _language_tag2locale
|
||||||
|
{
|
||||||
|
my $langtags = $_[0];
|
||||||
|
my @locales;
|
||||||
|
foreach my $maybe (split /[\n\r\t ,]+/, $langtags)
|
||||||
|
{
|
||||||
|
push @locales,
|
||||||
|
lc($1) . ( $2 ? ('_' . uc($2)) : '' )
|
||||||
|
if $maybe =~ m/^([a-zA-Z]{2})(?:-([a-zA-Z]{2}))?$/s;
|
||||||
|
}
|
||||||
|
return $locales[0] unless wantarray; # might be undef!
|
||||||
|
return @locales; # might be empty!
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 availableLanguages()
|
||||||
|
|
||||||
|
Returns an array containing the available languages supported by the
|
||||||
|
server.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$ENV{ESMITH_CONFIG_DB}="10e-smith-lib/sysconfig-en_US.conf";
|
||||||
|
$ENV{ESMITH_I18N_ESMITHLOCALEDIR}="10e-smith-lib/etc/e-smith/locale";
|
||||||
|
|
||||||
|
my $i18n = new esmith::I18N;
|
||||||
|
my @locales = grep !/CVS/, sort $i18n->availableLanguages;
|
||||||
|
|
||||||
|
is_deeply(\@locales, [('en-us', 'es', 'fr-ca', 'jk', 'wx-yz')], "Locales match" );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub availableLanguages
|
||||||
|
{
|
||||||
|
my ($self) = shift;
|
||||||
|
|
||||||
|
my $localedir = $ENV{ESMITH_I18N_ESMITHLOCALEDIR} || '/etc/e-smith/locale';
|
||||||
|
|
||||||
|
return () unless opendir LOCALE, $localedir;
|
||||||
|
|
||||||
|
my @locales = grep(!/\./, readdir LOCALE);
|
||||||
|
closedir LOCALE;
|
||||||
|
return @locales;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 fallbackLanguage()
|
||||||
|
|
||||||
|
Return system fallback language
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub fallbackLanguage()
|
||||||
|
{
|
||||||
|
return "en-us";
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 preferredLanguage()
|
||||||
|
|
||||||
|
Returns the preferred language, determined by the HTTP_ACCEPT_LANGUAGE
|
||||||
|
setting from the browser and the available languages on the server.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $i18n = new esmith::I18N;
|
||||||
|
delete $ENV{HTTP_ACCEPT_LANGUAGE};
|
||||||
|
|
||||||
|
is( $i18n->preferredLanguage(), "en-us", "Preferred language is en-us");
|
||||||
|
is( $i18n->preferredLanguage("en-us"), "en-us", "Preferred language is en-us");
|
||||||
|
is( $i18n->preferredLanguage("en-us, fr-ca"), "en-us", "Preferred language is en-us");
|
||||||
|
is( $i18n->preferredLanguage("fr-ca, en-us"), "fr-ca", "Preferred language is fr-ca");
|
||||||
|
|
||||||
|
$ENV{HTTP_ACCEPT_LANGUAGE} = "de, es";
|
||||||
|
is( $i18n->preferredLanguage(), "es", "Preferred language is es");
|
||||||
|
|
||||||
|
$ENV{HTTP_ACCEPT_LANGUAGE} = "de, fr-ca, es, en-us";
|
||||||
|
is( $i18n->preferredLanguage(), "fr-ca", "Preferred language is fr-ca");
|
||||||
|
|
||||||
|
$ENV{HTTP_ACCEPT_LANGUAGE} = "de, es, fr-ca, en-us";
|
||||||
|
is( $i18n->preferredLanguage(), "es", "Preferred language is es");
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub preferredLanguage
|
||||||
|
{
|
||||||
|
my ($self, $browser_languages) = @_;
|
||||||
|
|
||||||
|
$browser_languages ||= $ENV{HTTP_ACCEPT_LANGUAGE}
|
||||||
|
||= $self->fallbackLanguage;
|
||||||
|
|
||||||
|
my @availableLanguages = $self->availableLanguages;
|
||||||
|
|
||||||
|
my $acceptor = I18N::AcceptLanguage->new();
|
||||||
|
my $language = $acceptor->accepts($browser_languages, \@availableLanguages)
|
||||||
|
|| $self->fallbackLanguage;
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
77
root/usr/share/perl5/vendor_perl/esmith/Logger.pm
Normal file
77
root/usr/share/perl5/vendor_perl/esmith/Logger.pm
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::Logger;
|
||||||
|
|
||||||
|
use Sys::Syslog qw(:DEFAULT setlogsock);
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::Logger - A filehandle abstraction around Syslog.
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::Logger;
|
||||||
|
|
||||||
|
tie *FH, 'esmith::Logger';
|
||||||
|
print FH "log message";
|
||||||
|
close FH;
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
our $VERSION = sprintf '%d.%03d', q$Revision: 1.100 $ =~ /: (\d+).(\d+)/;
|
||||||
|
|
||||||
|
sub TIEHANDLE
|
||||||
|
{
|
||||||
|
my $class = ref($_[0]) || $_[0]; shift;
|
||||||
|
my $self;
|
||||||
|
my $title = shift || 'e-smith';
|
||||||
|
setlogsock 'unix';
|
||||||
|
openlog($title, 'pid', 'local1');
|
||||||
|
return bless \$self, $class;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub PRINT
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
syslog('info', "%s", "@_");
|
||||||
|
}
|
||||||
|
|
||||||
|
sub PRINTF
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $fmt = shift;
|
||||||
|
syslog('info', $fmt, @_);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub WRITE
|
||||||
|
{
|
||||||
|
die "Sorry, WRITE unimplemented.\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
sub READ
|
||||||
|
{
|
||||||
|
die "Can't read from logger.\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
sub READLINE
|
||||||
|
{
|
||||||
|
die "Can't read from logger.\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
sub GETC
|
||||||
|
{
|
||||||
|
die "Can't read from logger.\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
sub CLOSE
|
||||||
|
{
|
||||||
|
closelog();
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
16
root/usr/share/perl5/vendor_perl/esmith/NavigationDB.pm
Normal file
16
root/usr/share/perl5/vendor_perl/esmith/NavigationDB.pm
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2008 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::NavigationDB;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
|
||||||
|
use esmith::ConfigDB::UTF8;
|
||||||
|
our @ISA = qw( esmith::ConfigDB::UTF8 );
|
||||||
|
|
||||||
|
1;
|
||||||
|
|
157
root/usr/share/perl5/vendor_perl/esmith/NetworksDB.pm
Normal file
157
root/usr/share/perl5/vendor_perl/esmith/NetworksDB.pm
Normal file
@ -0,0 +1,157 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::NetworksDB;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
|
||||||
|
use esmith::DB::db;
|
||||||
|
our @ISA = qw( esmith::DB::db );
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::NetworksDB - interface to esmith networks database
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::NetworksDB;
|
||||||
|
my $c = esmith::NetworksDB->open;
|
||||||
|
|
||||||
|
# everything else works just like esmith::DB::db
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module provides an abstracted interface to the esmith master
|
||||||
|
configuration database.
|
||||||
|
|
||||||
|
Unless otherwise noted, esmith::NetworksDB acts like esmith::DB::db.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
=head2 open()
|
||||||
|
|
||||||
|
Like esmith::DB->open, but if given no $file it will try to open the
|
||||||
|
file in the ESMITH_NETWORKS_DB environment variable or networks.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use_ok("esmith::NetworksDB");
|
||||||
|
|
||||||
|
$C = esmith::NetworksDB->open('10e-smith-lib/networks.conf');
|
||||||
|
isa_ok($C, 'esmith::NetworksDB');
|
||||||
|
is( $C->get("10.0.0.0")->prop('Mask'), "255.255.255.0",
|
||||||
|
"We can get stuff from the db");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
$file = $file || $ENV{ESMITH_NETWORKS_DB} || "networks";
|
||||||
|
return $class->SUPER::open($file);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 open_ro()
|
||||||
|
|
||||||
|
Like esmith::DB->open_ro, but if given no $file it will try to open the
|
||||||
|
file in the ESMITH_NETWORKS_DB environment variable or networks.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open_ro
|
||||||
|
{
|
||||||
|
my ( $class, $file ) = @_;
|
||||||
|
$file = $file || $ENV{ESMITH_NETWORKS_DB} || "networks";
|
||||||
|
return $class->SUPER::open_ro($file);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 networks
|
||||||
|
|
||||||
|
Return a list of all objects of type "network".
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub networks {
|
||||||
|
my ($self) = @_;
|
||||||
|
return $self->get_all_by_prop(type => 'network');
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 local_access_spec ([$access])
|
||||||
|
|
||||||
|
Compute the network/netmask entries which are to treated as local access.
|
||||||
|
|
||||||
|
There is also an optional access parameter which can further restrict
|
||||||
|
the values returned. If C<access> is C<localhost>, this routine will only
|
||||||
|
return a single value, equating to access from localhost only.
|
||||||
|
|
||||||
|
If called in scalar context, the returned string is suitable for
|
||||||
|
use in /etc/hosts.allow, smb.conf and httpd.conf, for example:
|
||||||
|
|
||||||
|
127.0.0.1 192.168.1.1/255.255.255.0
|
||||||
|
|
||||||
|
Note: The elements are space separated, which is suitable for use in
|
||||||
|
hosts.allow, smb.conf and httpd.conf. httpd.conf does not permit
|
||||||
|
comma separated lists in C<allow from> directives. Each element is either
|
||||||
|
an IP address, or a network/netmask string.
|
||||||
|
|
||||||
|
If called in list context, returns the array of addresses and network/netmask
|
||||||
|
strings. It's trivial, of course, to convert an array to a comma separated
|
||||||
|
list :-)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub local_access_spec
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $access = shift || "private";
|
||||||
|
|
||||||
|
my @localAccess = ("127.0.0.1");
|
||||||
|
|
||||||
|
if ( $access eq "localhost" )
|
||||||
|
{
|
||||||
|
# Nothing more to do
|
||||||
|
}
|
||||||
|
elsif ( $access eq "private" )
|
||||||
|
{
|
||||||
|
foreach my $network ( $self->networks )
|
||||||
|
{
|
||||||
|
my $element = $network->key;
|
||||||
|
my $mask = $network->prop('Mask');
|
||||||
|
$element .= "/$mask" unless ($mask eq "255.255.255.255");
|
||||||
|
push @localAccess, $element;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
elsif ( $access eq "public" )
|
||||||
|
{
|
||||||
|
@localAccess = ("ALL");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
warn "local_access_spec: unknown access value $access\n";
|
||||||
|
}
|
||||||
|
return wantarray ? @localAccess : "@localAccess";
|
||||||
|
}
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
SME Server Developers <bugs@e-smith.com>
|
||||||
|
|
||||||
|
=head1 SEE ALSO
|
||||||
|
|
||||||
|
L<esmith::DB::db>
|
||||||
|
|
||||||
|
L<esmith::DB::Record>
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
480
root/usr/share/perl5/vendor_perl/esmith/cgi.pm
Normal file
480
root/usr/share/perl5/vendor_perl/esmith/cgi.pm
Normal file
@ -0,0 +1,480 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::cgi;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use esmith::config;
|
||||||
|
use esmith::db;
|
||||||
|
use esmith::util;
|
||||||
|
|
||||||
|
BEGIN
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::cgi - Useful CGI routines for e-smith server and gateway
|
||||||
|
|
||||||
|
=head1 VERSION
|
||||||
|
|
||||||
|
This file documents C<esmith::cgi> version B<1.4.0>
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::cgi;
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module contains a collection of useful routines for working with
|
||||||
|
the e-smith manager's CGI interface.
|
||||||
|
=head1 WEB PAGE HEADER GENERATION ROUTINES
|
||||||
|
|
||||||
|
=head2 genHeaderNonCacheable($q, $confref, $title)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genHeaderNonCacheable
|
||||||
|
{
|
||||||
|
my ($q, $confref, $title) = @_;
|
||||||
|
genHeader ($q, $confref, $title, '-20y', 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genHeaderCacheableNoPasswordCheck($q, $confref, $title)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genHeaderCacheableNoPasswordCheck
|
||||||
|
{
|
||||||
|
my ($q, $confref, $title) = @_;
|
||||||
|
genHeader ($q, $confref, $title, '+1d', 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genHeaderCacheableNoPasswordCheck($q, $confref, $title)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genHeaderNonCacheableNoPasswordCheck
|
||||||
|
{
|
||||||
|
my ($q, $confref, $title) = @_;
|
||||||
|
genHeader ($q, $confref, $title, '-20y', 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genHeader($q, $confref, $title, $expiry, $checkpassword)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genHeader
|
||||||
|
{
|
||||||
|
my ($q, $confref, $title, $expiry, $checkpassword) = @_;
|
||||||
|
|
||||||
|
print $q->header (-EXPIRES => $expiry, charset => 'UTF-8');
|
||||||
|
|
||||||
|
genHeaderStartHTML ($q, "panel_main");
|
||||||
|
|
||||||
|
print $q->h1 ($title);
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genNavigationHeader($q)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genNavigationHeader
|
||||||
|
{
|
||||||
|
my ($q, $num) = @_;
|
||||||
|
|
||||||
|
print $q->header (-EXPIRES => '-20y', charset => 'UTF-8');
|
||||||
|
|
||||||
|
genHeaderStartHTML ($q, "panel_nav", $num);
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genNoframesHeader($q)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genNoframesHeader
|
||||||
|
{
|
||||||
|
my ($q) = @_;
|
||||||
|
|
||||||
|
print $q->header (-EXPIRES => '-20y', charset => 'UTF-8');
|
||||||
|
genHeaderStartHTML ($q, "panel_main");
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genHeaderStartHTML($q)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genHeaderStartHTML
|
||||||
|
{
|
||||||
|
my ($q, $page_type, $num) = @_;
|
||||||
|
my ($cssFile);
|
||||||
|
my ($bodyStyle);
|
||||||
|
my ($script) = "//This swaps the class of the selected item.\n"
|
||||||
|
."function swapClass(){\n"
|
||||||
|
."var i,x,tB,j=0,tA=new Array(),arg=swapClass.arguments;\n"
|
||||||
|
."if(document.getElementsByTagName){for(i=4;i<arg.length;i++){tB=document.getElementsByTagName(arg[i]);\n"
|
||||||
|
."for(x=0;x<tB.length;x++){tA[j]=tB[x];j++;}}for(i=0;i<tA.length;i++){\n"
|
||||||
|
."if(tA[i].className){if(tA[i].id==arg[1]){if(arg[0]==1){\n"
|
||||||
|
."tA[i].className=(tA[i].className==arg[3])?arg[2]:arg[3];}else{tA[i].className=arg[2];}\n"
|
||||||
|
."}else if(arg[0]==1 && arg[1]=='none'){if(tA[i].className==arg[2] || tA[i].className==arg[3]){\n"
|
||||||
|
."tA[i].className=(tA[i].className==arg[3])?arg[2]:arg[3];}\n"
|
||||||
|
."}else if(tA[i].className==arg[2]){tA[i].className=arg[3];}}}}}\n";
|
||||||
|
|
||||||
|
if ($page_type eq "panel_nav") {
|
||||||
|
$cssFile = "sme_menu.css";
|
||||||
|
$bodyStyle = "menu"
|
||||||
|
}
|
||||||
|
elsif ($page_type eq "panel_main") {
|
||||||
|
$cssFile = "sme_main.css";
|
||||||
|
$bodyStyle = "main"
|
||||||
|
}
|
||||||
|
# the -CLASS thing gets sent as a body class, not in the header
|
||||||
|
print $q->start_html (-TITLE => 'SME Server server manager',
|
||||||
|
-META => {'copyright' => 'Copyright 1999-2006 Mitel Networks Corporation, Copyright (C) ____COPYYEARS____ Koozali Foundation, Inc.'},
|
||||||
|
-SCRIPT => "$script",
|
||||||
|
-CLASS => "$bodyStyle",
|
||||||
|
-STYLE => {
|
||||||
|
-code => '@import url("/server-common/css/'.$cssFile.'");',
|
||||||
|
-src => '/server-common/css/sme_core.css'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 WEB PAGE FOOTER GENERATION ROUTINES
|
||||||
|
|
||||||
|
=head2 genFooter($q)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genFooter
|
||||||
|
{
|
||||||
|
my ($q) = @_;
|
||||||
|
|
||||||
|
if ($q->isa('CGI::FormMagick'))
|
||||||
|
{
|
||||||
|
print $q->parse_template("/etc/e-smith/web/common/foot.tmpl");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $release = esmith::util::determineRelease();
|
||||||
|
|
||||||
|
print $q->p
|
||||||
|
($q->hr ({-CLASS => "sme-copyrightbar"}),
|
||||||
|
$q->div ({-CLASS => "sme-copyright"},
|
||||||
|
"SME Server server ${release}<BR>" .
|
||||||
|
"Copyright 1999-2006 Mitel Networks Corporation, Copyright (C) ____COPYYEARS____ Koozali Foundation, Inc..<BR>" .
|
||||||
|
"All rights reserved.")
|
||||||
|
);
|
||||||
|
|
||||||
|
print '</DIV>';
|
||||||
|
print $q->end_html;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genFooterNoCopyright($q)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genFooterNoCopyright
|
||||||
|
{
|
||||||
|
my ($q) = @_;
|
||||||
|
print $q->p ($q->hr);
|
||||||
|
print $q->end_html;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genNavigationFooter($q)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genNavigationFooter
|
||||||
|
{
|
||||||
|
my ($q) = @_;
|
||||||
|
print $q->end_html;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genNoframesFooter($q)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genNoframesFooter
|
||||||
|
{
|
||||||
|
my ($q) = @_;
|
||||||
|
print $q->end_html;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 FONT ROUTINES
|
||||||
|
|
||||||
|
=head2 curFont()
|
||||||
|
|
||||||
|
Returns the preferred font faces eg. "Verdana, Arial, Helvetica, sans-serif".
|
||||||
|
This should be done by CSS now, so if you're calling this, you shouldn't be.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub curFont
|
||||||
|
{
|
||||||
|
return "Verdana, Arial, Helvetica, sans-serif";
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 TABLE GENERATION ROUTINES
|
||||||
|
|
||||||
|
=head2 genCell($q, $text)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genCell
|
||||||
|
{
|
||||||
|
my ($q, $text, $class) = @_;
|
||||||
|
|
||||||
|
if ($text =~ /^\s*$/){$text = " "}
|
||||||
|
if ($class) { return $q->td({-class => "$class"}, $text),"\n";}
|
||||||
|
else { return $q->td ($text),"\n";}
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genDoubleCell($q, $text);
|
||||||
|
|
||||||
|
Generates a cell which spans two columns, containing the text specified.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genDoubleCell
|
||||||
|
{
|
||||||
|
my ($q, $text) = @_;
|
||||||
|
if ($text =~ /^\s*$/){ $text = " " }
|
||||||
|
return $q->td ({colspan => 2}, $text),"\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genSmallCell($q, $text, $type, $colspan)
|
||||||
|
|
||||||
|
Generates a cell with "small" text (font size is 80%).
|
||||||
|
"$type" can be one of:
|
||||||
|
"normal" : creates <td class="sme-border"> cell
|
||||||
|
"header" : creates <th class="sme-border"> cell
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genSmallCell
|
||||||
|
{
|
||||||
|
my ($q, $text, $type, $colspan) = @_;
|
||||||
|
$text = '' unless defined $text;
|
||||||
|
$type ||= 'normal';
|
||||||
|
$colspan ||= 1;
|
||||||
|
if ($text =~ /^\s*$/){ $text = " " }
|
||||||
|
if ("$type" eq "header") {
|
||||||
|
return $q->th ({class=>"sme-border", colspan=>$colspan}, $text)."\n";
|
||||||
|
} else {
|
||||||
|
return $q->td ({class=>"sme-border", colspan=>$colspan}, $text)."\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genSmallCellCentered($q, $text)
|
||||||
|
|
||||||
|
Generates a cell with "small" text (font size is 80%), centered.
|
||||||
|
creates <td class="sme-border-center"> cell
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genSmallCellCentered
|
||||||
|
{
|
||||||
|
my ($q, $text) = @_;
|
||||||
|
if ($text =~ /^\s*$/){ $text = " " }
|
||||||
|
return $q->td ({class => "sme-border-center"}, $text)."\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genSmallCellRightJustified($q, $text)
|
||||||
|
|
||||||
|
=head2 genSmallCellCentered($q, $text)
|
||||||
|
|
||||||
|
Generates a cell with "small" text (font size is 80%), right justified.
|
||||||
|
creates <td class="sme-border-right"> cell
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genSmallCellRightJustified
|
||||||
|
{
|
||||||
|
my ($q, $text) = @_;
|
||||||
|
if ($text =~ /^\s*$/){ $text = " " }
|
||||||
|
return $q->td ({class => "sme-border-right"}, $text)."\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genSmallRedCell($q, $text)
|
||||||
|
|
||||||
|
Generates a cell with "small" text (font size is 80%), left justified.
|
||||||
|
creates <td class="sme-border-warning"> cell
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genSmallRedCell
|
||||||
|
{
|
||||||
|
my ($q, $text) = @_;
|
||||||
|
if ($text =~ /^\s*$/){ $text = " " }
|
||||||
|
return $q->td ({class => "sme-border-warning"}, $text)."\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genTextRow($q, $text)
|
||||||
|
|
||||||
|
Returns a table row containing a two-column cell containing $text.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genTextRow
|
||||||
|
{
|
||||||
|
my ($q, $text) = @_;
|
||||||
|
if ($text =~ /^\s*$/){ $text = " " }
|
||||||
|
return "\n",$q->Tr ($q->td ({colspan => 2}, $text)),"\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genButtonRow($q, $button)
|
||||||
|
|
||||||
|
Returns a table row containing an empty first cell and a second cell
|
||||||
|
containing a button with the value $button.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genButtonRow
|
||||||
|
{
|
||||||
|
my ($q, $button) = @_;
|
||||||
|
|
||||||
|
# return $q->Tr ($q->td ({-class => "sme-submitbutton", -colspan => "2"},$q->b ($button))),"\n";
|
||||||
|
# return $q->Tr ($q->td (' '),
|
||||||
|
# $q->td ({-class => "sme-submitbutton"},$q->b ($button))),"\n";
|
||||||
|
return $q->Tr ({-class => "sme-layout"}, $q->th ({-class => "sme-layout", colspan => "2"},$q->b ($button))),"\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genNameValueRow($q, $fieldlabel, $fieldname, $fieldvalue)
|
||||||
|
|
||||||
|
Returns a table row with two cells. The first has the text
|
||||||
|
"$fieldlabel:" in it, and the second has a text field with the default
|
||||||
|
value $fieldvalue and the name $fieldname.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genNameValueRow
|
||||||
|
{
|
||||||
|
my ($q, $fieldlabel, $fieldname, $fieldvalue) = @_;
|
||||||
|
|
||||||
|
return $q->Tr (
|
||||||
|
$q->td ({-class => "sme-noborders-label"},
|
||||||
|
"$fieldlabel:"),"\n",
|
||||||
|
$q->td ({-class => "sme-noborders-content"},
|
||||||
|
$q->textfield (
|
||||||
|
-name => $fieldname,
|
||||||
|
-override => 1,
|
||||||
|
-default => $fieldvalue,
|
||||||
|
-size => 32))),"\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
sub genWidgetRow($q, $fieldlabel, $popup)
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
# used only by backup panel as far as I can see
|
||||||
|
sub genWidgetRow
|
||||||
|
{
|
||||||
|
my ($q, $fieldlabel, $popup) = @_;
|
||||||
|
|
||||||
|
return $q->Tr ($q->td ("$fieldlabel:"),
|
||||||
|
$q->td ($popup));
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 STATUS AND ERROR REPORT GENERATION ROUTINES
|
||||||
|
|
||||||
|
=head2 genResult($q, $msg)
|
||||||
|
|
||||||
|
Generates a "status report" page, including the footer
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genResult
|
||||||
|
{
|
||||||
|
my ($q, $msg) = @_;
|
||||||
|
|
||||||
|
print $q->p ($msg);
|
||||||
|
genFooter ($q);
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 genStateError($q, $confref)
|
||||||
|
|
||||||
|
Subroutine to generate "unknown state" error message.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub genStateError
|
||||||
|
{
|
||||||
|
my ($q, $confref) = @_;
|
||||||
|
|
||||||
|
genHeaderNonCacheable ($q, $confref, "Internal error");
|
||||||
|
genResult ($q, "Internal error! Unknown state: " . $q->param ("state") . ".");
|
||||||
|
}
|
||||||
|
|
||||||
|
END
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# return "1" to make the import process return success
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
1;
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
Mitel Networks Corporation
|
||||||
|
|
||||||
|
For more information, see http://e-smith.org/
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
640
root/usr/share/perl5/vendor_perl/esmith/config.pm
Normal file
640
root/usr/share/perl5/vendor_perl/esmith/config.pm
Normal file
@ -0,0 +1,640 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::config;
|
||||||
|
use strict;
|
||||||
|
use vars qw($VERSION);
|
||||||
|
$VERSION = 1.45;
|
||||||
|
|
||||||
|
use Sys::Syslog qw(:DEFAULT setlogsock);
|
||||||
|
use Fcntl qw(:DEFAULT :flock);
|
||||||
|
use Carp qw(cluck);
|
||||||
|
|
||||||
|
my $Default_Config = '/home/e-smith/db/configuration';
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::config - Access e-smith config files via hashes
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::config;
|
||||||
|
|
||||||
|
my %config;
|
||||||
|
my $config_obj = tie %config, 'esmith::config', $config_file;
|
||||||
|
|
||||||
|
# Read in the value of Wibble from the $config_file.
|
||||||
|
print $config{Wibble};
|
||||||
|
|
||||||
|
# Write out the value of Wibble to the $config_file.
|
||||||
|
$config{Wibble} = 42;
|
||||||
|
|
||||||
|
my $filename = $config_obj->filename;
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
The esmith::config package enables Perl programs to read and write
|
||||||
|
entries from the e-smith configuration file using a simple hash
|
||||||
|
interface.
|
||||||
|
|
||||||
|
The configuration file has a simple ASCII representation,
|
||||||
|
with one "key=value" entry per line.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use_ok('esmith::config');
|
||||||
|
chdir '10e-smith-lib';
|
||||||
|
|
||||||
|
%Expect = ( foo => 'bar',
|
||||||
|
'this key' => 'has whitespace',
|
||||||
|
'that key ' => 'has trailing whitespace',
|
||||||
|
' another key' => 'has leading whitespace',
|
||||||
|
'this value' => ' has leading whitespace',
|
||||||
|
'that value' => 'has trailing whitespace ',
|
||||||
|
'tricky value' => 'with=equals.',
|
||||||
|
);
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=head2 Tying
|
||||||
|
|
||||||
|
tie %config, 'esmith::config', $config_file;
|
||||||
|
|
||||||
|
Reads in the configuration from the given $config_file, returning a
|
||||||
|
tied hash (%config) populated with the keys & values from the
|
||||||
|
$config_file which you can then use like a normal hash. Any writes or
|
||||||
|
deletes are immediately written back to the $config_file.
|
||||||
|
|
||||||
|
If no $config_file is supplied it falls back to the environment variable
|
||||||
|
ESMITH_CONFIG_DB, and finally defaults to F</home/e-smith/db/configuration>
|
||||||
|
|
||||||
|
If the $config_file doesn't exist it will create one for you.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my %config;
|
||||||
|
tie %config, 'esmith::config', 'dummy.conf';
|
||||||
|
ok( tied %config, 'tie worked' );
|
||||||
|
is( $config{foo}, 'bar', ' theres stuff in it' );
|
||||||
|
ok( !exists $config{FILENAME}, ' it only contains config info');
|
||||||
|
is( tied(%config)->{FILENAME}, 'dummy.conf',
|
||||||
|
' and the real object is inside');
|
||||||
|
|
||||||
|
tie %config, 'esmith::config', 'I_dont_exist';
|
||||||
|
ok( tied %config, 'tying a non-existant file' );
|
||||||
|
is( keys %config, 0, ' and its empty' );
|
||||||
|
$config{foo} = 42;
|
||||||
|
isnt( -s 'I_dont_exist', 0 );
|
||||||
|
untie %config;
|
||||||
|
ok( unlink 'I_dont_exist' );
|
||||||
|
|
||||||
|
$ENV{ESMITH_CONFIG_DB} = "dummy.conf";
|
||||||
|
tie %config, 'esmith::config';
|
||||||
|
ok( tied %config, 'tie to ESMITH_CONFIG_DB worked' );
|
||||||
|
is_deeply(\%config, \%Expect, " picked up data");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=head2 Methods
|
||||||
|
|
||||||
|
You can get at the underlying esmith::config object by using tied().
|
||||||
|
|
||||||
|
my $config_obj = tied %config;
|
||||||
|
|
||||||
|
it has a few useful methods.
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item filename
|
||||||
|
|
||||||
|
my $file = $config_obj->filename;
|
||||||
|
|
||||||
|
Gets the config filename this object is tied to.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my %config;
|
||||||
|
my $obj = tie %config, 'esmith::config', 'dummy.conf';
|
||||||
|
is( $obj->filename, 'dummy.conf', 'filename()' );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub filename {
|
||||||
|
my($self) = shift;
|
||||||
|
return $self->{FILENAME};
|
||||||
|
}
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=begin _private
|
||||||
|
|
||||||
|
=head2 Private methods
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item _readconf
|
||||||
|
|
||||||
|
my $config = _readconf($config_file);
|
||||||
|
|
||||||
|
Returns a hash ref of config key/value pairs read out of the given
|
||||||
|
$config_file. If $config_file doesn't exist an empty hash ref will be
|
||||||
|
returned.
|
||||||
|
|
||||||
|
_readconf() understands the config file to be formatted as individual
|
||||||
|
lines of simply:
|
||||||
|
|
||||||
|
key=value
|
||||||
|
|
||||||
|
any further complexity of parsing the value is handled elsewhere.
|
||||||
|
|
||||||
|
=end _private
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $config = esmith::config::_readconf('dummy.conf');
|
||||||
|
isnt( keys %$config, 0, '_readconf() got something' );
|
||||||
|
is_deeply( $config, \%Expect, ' read in the right values' );
|
||||||
|
|
||||||
|
$config = esmith::config::_readconf('I_dont_exist');
|
||||||
|
isa_ok( $config, 'HASH', '_readconf from a non-existent file' );
|
||||||
|
is( keys %$config, 0, ' and its empty' );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _readconf
|
||||||
|
{
|
||||||
|
my ($self, $filename) = @_;
|
||||||
|
|
||||||
|
my %config = ();
|
||||||
|
|
||||||
|
unless (open (FH, $filename))
|
||||||
|
{
|
||||||
|
if (-f $filename)
|
||||||
|
{
|
||||||
|
&log("Config: ERROR: \"$filename\" exists but is not readable");
|
||||||
|
}
|
||||||
|
return \%config;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $binmode = $self->_read_binmode;
|
||||||
|
binmode(FH, $binmode) if $binmode;
|
||||||
|
while (my $line = <FH>)
|
||||||
|
{
|
||||||
|
chomp $line;
|
||||||
|
|
||||||
|
# BLIND UNTAINT! Much code wrongly depends on this and
|
||||||
|
# they should be moved away from it.
|
||||||
|
$line =~ /(.*)/;
|
||||||
|
$line = $1;
|
||||||
|
|
||||||
|
# ignore comments and blank lines
|
||||||
|
next if $line =~ /^\s*$/ || $line =~ /^\s*#/;
|
||||||
|
|
||||||
|
my($key, $value) = split /=/, $line, 2;
|
||||||
|
$config{$key} = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
close(FH);
|
||||||
|
|
||||||
|
return \%config;
|
||||||
|
}
|
||||||
|
|
||||||
|
=begin _private
|
||||||
|
|
||||||
|
=item _writeconf
|
||||||
|
|
||||||
|
my $success = _writeconf($config_file, \%config);
|
||||||
|
|
||||||
|
The given $config_file is overwritten using the entries in %config.
|
||||||
|
|
||||||
|
Returns whether or not the write succeded.
|
||||||
|
|
||||||
|
=end _private
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $scratch = 'scratch.conf';
|
||||||
|
ok( esmith::config::_writeconf($scratch, \%Expect),
|
||||||
|
'_writeconf() says it worked' );
|
||||||
|
is_deeply( esmith::config::_readconf($scratch), \%Expect,
|
||||||
|
' wrote the right things' );
|
||||||
|
unlink $scratch;
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _writeconf
|
||||||
|
{
|
||||||
|
my ($self, $filename, $config) = @_;
|
||||||
|
|
||||||
|
eval {
|
||||||
|
if (-f $filename && ! -r $filename)
|
||||||
|
{
|
||||||
|
die "'$filename' exists but is not readable\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
sysopen (FH, "$filename.$$", O_RDWR | O_CREAT, 0660)
|
||||||
|
or die "Cannot open $filename.$$: $!\n";
|
||||||
|
my $binmode = $self->_write_binmode;
|
||||||
|
binmode(FH, $binmode) if $binmode;
|
||||||
|
|
||||||
|
die "Error writing to $filename.$$: $!" unless
|
||||||
|
printf FH <<EOF, scalar localtime;
|
||||||
|
# DO NOT MODIFY THIS FILE.
|
||||||
|
# This file is automatically maintained by the Mitel Networks SME Server
|
||||||
|
# configuration software. Manually editing this file may put your
|
||||||
|
# system in an unknown state.
|
||||||
|
#
|
||||||
|
# updated: %s
|
||||||
|
EOF
|
||||||
|
|
||||||
|
foreach my $key (sort keys %$config)
|
||||||
|
{
|
||||||
|
print FH "$key=$config->{$key}\n"
|
||||||
|
or die "Error writing to $filename.$$: $!";
|
||||||
|
}
|
||||||
|
|
||||||
|
close (FH) or die "Error closing $filename.$$: $!";
|
||||||
|
|
||||||
|
rename("$filename.$$", $filename)
|
||||||
|
or die "Couldn't rename $filename.$$ to $filename: $!";
|
||||||
|
|
||||||
|
my $gid = getgrnam ('admin') || 0;
|
||||||
|
chown (0, $gid, $filename);
|
||||||
|
};
|
||||||
|
|
||||||
|
if($@) {
|
||||||
|
chomp $@;
|
||||||
|
&log($@);
|
||||||
|
&log("'$filename' will not be updated");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=begin _private
|
||||||
|
|
||||||
|
=item B<_lock_write>
|
||||||
|
|
||||||
|
=item B<_lock_read>
|
||||||
|
|
||||||
|
$self->_lock_write;
|
||||||
|
$self->_lock_read;
|
||||||
|
|
||||||
|
Sets up read (shared) or write (exclusive) locks on the config file.
|
||||||
|
This is actually locking a semaphore file.
|
||||||
|
|
||||||
|
Returns if the lock succeeded or failed.
|
||||||
|
|
||||||
|
=item B<_unlock>
|
||||||
|
|
||||||
|
$self->_unlock
|
||||||
|
|
||||||
|
Unlocks the config file.
|
||||||
|
|
||||||
|
=end _private
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _lock_write { $_[0]->_lock(LOCK_EX) }
|
||||||
|
sub _lock_read { $_[0]->_lock(LOCK_SH) }
|
||||||
|
|
||||||
|
sub _lock {
|
||||||
|
return if $] eq 5.006; # Locking is broken in perl 5.6.0
|
||||||
|
my($self, $lock) = @_;
|
||||||
|
|
||||||
|
my $semaphore = $self->{SEMAPHORE_FILE} = $self->{FILENAME}.'.lock';
|
||||||
|
|
||||||
|
eval {
|
||||||
|
open(my $fh, ">>$semaphore") or
|
||||||
|
die "Can't open '$semaphore' semaphore: $!";
|
||||||
|
$self->{SEMAPHORE} = $fh;
|
||||||
|
|
||||||
|
flock($fh, $lock) or
|
||||||
|
die "Can't lock '$semaphore' semaphore: $!";
|
||||||
|
};
|
||||||
|
if( $@ ) {
|
||||||
|
warn $@;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
sub _unlock {
|
||||||
|
return if $] eq 5.006; # Locking is broken in perl 5.6.0
|
||||||
|
my($self) = @_;
|
||||||
|
|
||||||
|
eval {
|
||||||
|
flock($self->{SEMAPHORE}, LOCK_UN) or
|
||||||
|
die "Can't unlock $self->{SEMAPHORE_FILE}: $!";
|
||||||
|
|
||||||
|
unlink $self->{SEMAPHORE_FILE};
|
||||||
|
delete $self->{SEMAPHORE_FILE};
|
||||||
|
delete $self->{SEMAPHORE};
|
||||||
|
};
|
||||||
|
if( $@ ) {
|
||||||
|
warn $@;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=end _private
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# Constructor for the tied hash. If filename not specified,
|
||||||
|
# defaults to '/home/e-smith/db/configuration'.
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
sub TIEHASH
|
||||||
|
{
|
||||||
|
my $class = shift;
|
||||||
|
my $filename = shift || $ENV{ESMITH_CONFIG_DB} || $Default_Config;
|
||||||
|
|
||||||
|
if ($filename =~ m:^/home/e-smith/\w+$: )
|
||||||
|
{
|
||||||
|
cluck "*WARNING* esmith::config($filename) called with old " .
|
||||||
|
"database path. The following package needs to be updated: ";
|
||||||
|
|
||||||
|
$filename =~ s:e-smith:e-smith/db:;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $self =
|
||||||
|
{
|
||||||
|
FILENAME => $filename,
|
||||||
|
CONFIG => {},
|
||||||
|
};
|
||||||
|
bless $self, $class;
|
||||||
|
|
||||||
|
$self->{CONFIG} = $self->_readconf($filename);
|
||||||
|
|
||||||
|
return $self;
|
||||||
|
}
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# Look up a configuration parameter.
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
sub FETCH
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $key = shift;
|
||||||
|
|
||||||
|
# Trim leading and trailing whitespace from the key.
|
||||||
|
$key =~ s/^\s+|\s+$//g;
|
||||||
|
|
||||||
|
return $self->{CONFIG}{$key};
|
||||||
|
}
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# Store a configuration parameter.
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
sub STORE
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $key = shift;
|
||||||
|
my $value = shift;
|
||||||
|
|
||||||
|
die "key not defined" unless defined $key;
|
||||||
|
die "value not defined for key $key" unless defined $value;
|
||||||
|
|
||||||
|
if( $value =~ /\n/ or $key =~ /\n/ ) {
|
||||||
|
&log("$self->{FILENAME}: esmith::config doesn't support newlines in ".
|
||||||
|
"keys or values. Truncating.");
|
||||||
|
$key =~ s/\n.*//s;
|
||||||
|
$value =~ s/\n.*//s;
|
||||||
|
}
|
||||||
|
|
||||||
|
# Trim leading and trailing whitespace from the key and value.
|
||||||
|
$key =~ s/^\s+|\s+$//g;
|
||||||
|
$value =~ s/^\s+|\s+$//g;
|
||||||
|
|
||||||
|
# Make sure that the value has a type. Given the format, it should be
|
||||||
|
# sufficient to ensure that it does not begin with a pipe char.
|
||||||
|
if ($value =~ /^\|/)
|
||||||
|
{
|
||||||
|
warn "ERROR: You should not set a config record without a type (key was $key).\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
# read in config again, just in case it changed
|
||||||
|
$self->_lock_write;
|
||||||
|
$self->{CONFIG} = $self->_readconf($self->{FILENAME});
|
||||||
|
|
||||||
|
if (exists $self->{CONFIG}{$key} and
|
||||||
|
$self->{CONFIG}{$key} eq $value)
|
||||||
|
{
|
||||||
|
$self->_unlock;
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $msg = "$self->{FILENAME}: OLD $key=";
|
||||||
|
|
||||||
|
if (exists $self->{CONFIG}{$key})
|
||||||
|
{
|
||||||
|
$msg .= "$self->{CONFIG}{$key}";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$msg .= "(undefined)";
|
||||||
|
}
|
||||||
|
|
||||||
|
&log($msg);
|
||||||
|
|
||||||
|
$self->{CONFIG} {$key} = $value;
|
||||||
|
&log("$self->{FILENAME}: NEW $key=$self->{CONFIG}{$key}");
|
||||||
|
|
||||||
|
$self->_writeconf ($self->{FILENAME}, $self->{CONFIG});
|
||||||
|
$self->_unlock;
|
||||||
|
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# Delete a configuration parameter.
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
sub DELETE
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $key = shift;
|
||||||
|
|
||||||
|
# Trim leading and trailing whitespace from the key.
|
||||||
|
$key =~ s/^\s+|\s+$//g;
|
||||||
|
|
||||||
|
# read in config again, just in case it changed
|
||||||
|
$self->_lock_write;
|
||||||
|
$self->{CONFIG} = $self->_readconf($self->{FILENAME});
|
||||||
|
|
||||||
|
my $previous = delete $self->{CONFIG} {$key};
|
||||||
|
$self->_writeconf ($self->{FILENAME}, $self->{CONFIG});
|
||||||
|
$self->_unlock;
|
||||||
|
|
||||||
|
&log("$self->{FILENAME}: DELETE $key=$previous");
|
||||||
|
|
||||||
|
return $previous;
|
||||||
|
}
|
||||||
|
|
||||||
|
=begin _private
|
||||||
|
|
||||||
|
=item CLEAR
|
||||||
|
|
||||||
|
tie method: Clear the configuration file
|
||||||
|
|
||||||
|
=end _private
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
my $scratch = 'scratch.conf';
|
||||||
|
tie %config, 'esmith::config', $scratch;
|
||||||
|
ok( tied %config, 'tying a non-existant file' );
|
||||||
|
is( keys %config, 0, ' and its empty' );
|
||||||
|
$config{fibble} = 'blah';
|
||||||
|
isnt( keys %config, 0, ' and its not empty now' );
|
||||||
|
%config = ();
|
||||||
|
is( keys %config, 0, ' and CLEAR made it empty again' );
|
||||||
|
unlink $scratch;
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub CLEAR
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
|
||||||
|
$self->{CONFIG} = ();
|
||||||
|
$self->_writeconf ($self->{FILENAME}, $self->{CONFIG});
|
||||||
|
|
||||||
|
&log("$self->{FILENAME}: CLEAR");
|
||||||
|
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# Check whether a particular key exists in the configuration file.
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
sub EXISTS
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $key = shift;
|
||||||
|
|
||||||
|
# Trim leading and trailing whitespace from the key.
|
||||||
|
$key =~ s/^\s+|\s+$//g;
|
||||||
|
|
||||||
|
return exists $self->{CONFIG} {$key};
|
||||||
|
}
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# FIRSTKEY is called whenever we start iterating over the
|
||||||
|
# configuration table. We cache the configuration table at
|
||||||
|
# this point to ensure reasonable results if the
|
||||||
|
# configuration file is changed by another program during
|
||||||
|
# the iteration.
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
sub FIRSTKEY
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
|
||||||
|
my $discard = keys %{$self->{CONFIG}}; # reset each() iterator
|
||||||
|
|
||||||
|
return each %{$self->{CONFIG}};
|
||||||
|
}
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# NEXTKEY is called for all iterations after the first. We
|
||||||
|
# just keep returning results from the cached configuration
|
||||||
|
# table. A null array is returned at the end. If the caller
|
||||||
|
# starts a new iteration, the FIRSTKEY subroutine is called
|
||||||
|
# again, causing the cache to be reloaded.
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
sub NEXTKEY
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
return each %{$self->{CONFIG}};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# Log messages to syslog
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
sub log
|
||||||
|
{
|
||||||
|
# There is a bug in Perl 5.00504 and above. If you are using the unix
|
||||||
|
# domain socket, do NOT use ndelay as part of the second argument
|
||||||
|
# to openlog().
|
||||||
|
|
||||||
|
my $msg = shift;
|
||||||
|
$msg =~ s/[^[:ascii:]]/_/g;
|
||||||
|
my $program = $0;
|
||||||
|
|
||||||
|
setlogsock 'unix';
|
||||||
|
openlog($program, 'pid', 'local1');
|
||||||
|
syslog('info', "%s", $msg);
|
||||||
|
closelog();
|
||||||
|
}
|
||||||
|
|
||||||
|
=item _read_binmode
|
||||||
|
|
||||||
|
return undef, indicating that by default binmode() need not be called after
|
||||||
|
file open.
|
||||||
|
|
||||||
|
=end _private
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _read_binmode
|
||||||
|
{
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _write_binmode
|
||||||
|
{
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head1 BUGS and CAVEATS
|
||||||
|
|
||||||
|
You can't have newlines in keys or values.
|
||||||
|
|
||||||
|
While the config values happen to be untainted B<do not depend on this
|
||||||
|
behavior> as it will change in the future.
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
SME Server Developers <bugs@e-smith.com>
|
||||||
|
|
||||||
|
For more information, see http://www.e-smith.org/
|
||||||
|
|
||||||
|
=head1 SEE ALSO
|
||||||
|
|
||||||
|
esmith::db
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
25
root/usr/share/perl5/vendor_perl/esmith/config/utf8.pm
Normal file
25
root/usr/share/perl5/vendor_perl/esmith/config/utf8.pm
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2008 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::config::utf8;
|
||||||
|
use warnings;
|
||||||
|
use strict;
|
||||||
|
|
||||||
|
use vars qw(@ISA);
|
||||||
|
@ISA = qw(esmith::config);
|
||||||
|
|
||||||
|
sub _read_binmode
|
||||||
|
{
|
||||||
|
return ":encoding(UTF-8)";
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _write_binmode
|
||||||
|
{
|
||||||
|
return ":utf8";
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
||||||
|
|
603
root/usr/share/perl5/vendor_perl/esmith/console.pm
Executable file
603
root/usr/share/perl5/vendor_perl/esmith/console.pm
Executable file
@ -0,0 +1,603 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2006 Mitel Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::console - A class to provide a backend library to the server console.
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::console;
|
||||||
|
|
||||||
|
my $console = esmith::console->new();
|
||||||
|
|
||||||
|
($rc, $choice) = $console->message_page
|
||||||
|
(
|
||||||
|
title => gettext("Administrator password not set"),
|
||||||
|
text => gettext("Sorry, you must set the administrator password."),
|
||||||
|
);
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This class provides a backend library of methods for the frontend console on
|
||||||
|
the server. The intent is that all of the whiptail code is hidden in this
|
||||||
|
library, and the frontend can just concern itself with the logical progression
|
||||||
|
through any and all applicable screens.
|
||||||
|
|
||||||
|
=head1 Methods
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
package esmith::console;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use vars qw($VERSION @ISA @EXPORT_OK);
|
||||||
|
use esmith::util;
|
||||||
|
use Locale::gettext;
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
use esmith::I18N;
|
||||||
|
|
||||||
|
@ISA = qw(Exporter);
|
||||||
|
|
||||||
|
use constant SCREEN_ROWS => 22;
|
||||||
|
use constant SCREEN_COLUMNS => 76;
|
||||||
|
use constant CONSOLE_SCREENS => "/sbin/e-smith/console-screens";
|
||||||
|
|
||||||
|
BEGIN
|
||||||
|
{
|
||||||
|
# disable CTRL-C
|
||||||
|
$SIG{INT} = 'IGNORE';
|
||||||
|
|
||||||
|
# Set PATH explicitly and clear related environment variables so that calls
|
||||||
|
# to external programs do not cause results to be tainted. See
|
||||||
|
# "perlsec" manual page for details.
|
||||||
|
|
||||||
|
$ENV {'PATH'} = '/sbin:/bin:/usr/sbin:/usr/bin:/usr/local/sbin:/usr/local/bin';
|
||||||
|
$ENV {'SHELL'} = '/bin/bash';
|
||||||
|
delete $ENV {'ENV'};
|
||||||
|
delete $ENV {'BASH_ENV'};
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 new
|
||||||
|
|
||||||
|
This is the class constructor.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub new
|
||||||
|
{
|
||||||
|
my $class = ref($_[0]) || $_[0];
|
||||||
|
my $self = {};
|
||||||
|
esmith::util::setRealToEffective ();
|
||||||
|
|
||||||
|
my $i18n = new esmith::I18N;
|
||||||
|
$i18n->setLocale("server-console");
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# Set stdin, stdout and stderr to console
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
if (defined $ARGV [0])
|
||||||
|
{
|
||||||
|
$ARGV[0] =~ /(console|tty\d*)/ && -c "/dev/$1"
|
||||||
|
or die gettext("Bad ttyname:"), " ", $ARGV[0], "\n";
|
||||||
|
my $tty = $1;
|
||||||
|
|
||||||
|
open (STDIN, "</dev/$tty") or die gettext("Can't redirect stdin"), ": $!\n";
|
||||||
|
open (STDOUT, ">/dev/$tty") or die gettext("Can't redirect stdout"), ": $!\n";
|
||||||
|
|
||||||
|
my $pid = open(STDERR, "|-");
|
||||||
|
die gettext("Can't fork"), ": $!\n" unless defined $pid;
|
||||||
|
|
||||||
|
unless ($pid)
|
||||||
|
{
|
||||||
|
exec qw(/usr/bin/logger -p local1.info -t console);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$self = bless $self, $class;
|
||||||
|
return $self;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 screen and dialog
|
||||||
|
|
||||||
|
These method are wrappers around whiptail and dialog, and permit the creation
|
||||||
|
of custom screens depending on the arguments passed. They are typically not
|
||||||
|
called directly, but are used by all of the other page methods that
|
||||||
|
follow. You should only call these method directly if none of the other
|
||||||
|
methods apply.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub screen
|
||||||
|
{
|
||||||
|
_screen(shift, "/usr/bin/dialog", @_);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub dialog
|
||||||
|
{
|
||||||
|
_screen(shift, "/usr/bin/dialog", @_);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub whiptail
|
||||||
|
{
|
||||||
|
_screen(shift, "/usr/bin/whiptail", @_);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _screen
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $whiptail = shift;
|
||||||
|
my @whiptailArgs = @_;
|
||||||
|
|
||||||
|
# now would be a good time to flush output buffers, so the partial
|
||||||
|
# buffers don't get copied:
|
||||||
|
|
||||||
|
$| = 1;
|
||||||
|
print "";
|
||||||
|
|
||||||
|
pipe (READER, WRITER)
|
||||||
|
or die gettext("Couldn't create pipe") . ": $!\n";
|
||||||
|
|
||||||
|
my $pid = fork;
|
||||||
|
|
||||||
|
if (! defined $pid)
|
||||||
|
{
|
||||||
|
die gettext("Couldn't fork") . ": $!\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
elsif ($pid == 0)
|
||||||
|
{
|
||||||
|
#----------------------------------------
|
||||||
|
# Child
|
||||||
|
#----------------------------------------
|
||||||
|
|
||||||
|
# Attach child's STDIN to the reading end of the pipe
|
||||||
|
close READER
|
||||||
|
or die gettext("Couldn't close reading end of pipe") , ": $!\n";
|
||||||
|
|
||||||
|
if ($whiptail =~ m{\bwhiptail$} ) {
|
||||||
|
# whiptail sends its output via STDERR. We temporarily
|
||||||
|
# shut off warnings so they don't interfere with that.
|
||||||
|
local $^W = 0;
|
||||||
|
|
||||||
|
open STDERR, ">& WRITER"
|
||||||
|
or die gettext("Couldn't connect STDERR to pipe"), ": $!\n";
|
||||||
|
|
||||||
|
close WRITER
|
||||||
|
or die gettext("Couldn't close writing end of pipe"), ": $!\n";
|
||||||
|
|
||||||
|
unshift @whiptailArgs, $whiptail,
|
||||||
|
'--backtitle', $self->backtitle;
|
||||||
|
} else {
|
||||||
|
use Fcntl qw/F_SETFD/;
|
||||||
|
|
||||||
|
# Clear close-on-exec on WRITER so that it stays open for dialog to use
|
||||||
|
fcntl(WRITER, F_SETFD, 0);
|
||||||
|
|
||||||
|
unshift @whiptailArgs, $whiptail,
|
||||||
|
'--backtitle', $self->backtitle, "--output-fd", fileno(WRITER);
|
||||||
|
}
|
||||||
|
exec @whiptailArgs;
|
||||||
|
die gettext("Couldn't exec:"), ": $!\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
#----------------------------------------
|
||||||
|
# Parent
|
||||||
|
#----------------------------------------
|
||||||
|
|
||||||
|
close WRITER;
|
||||||
|
|
||||||
|
my $choice = <READER>;
|
||||||
|
close READER;
|
||||||
|
|
||||||
|
waitpid ($pid, 0);
|
||||||
|
my $rc = $?;
|
||||||
|
|
||||||
|
return ($rc, $choice);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 backtitle
|
||||||
|
|
||||||
|
Console header line for each page
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub backtitle
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
|
||||||
|
my $db = esmith::ConfigDB->open_ro or die "Couldn't open ConfigDB\n";
|
||||||
|
|
||||||
|
sprintf("%-33s%45s",
|
||||||
|
($db->get_prop('sysconfig', 'ProductName') || "SME Server") . " " .
|
||||||
|
($db->get_prop('sysconfig', 'ReleaseVersion') || "UNKNOWN"),
|
||||||
|
"Copyright (C) 1999-2006 Mitel Corporation, Copyright (C) ____COPYYEARS____ Koozali Foundation, Inc."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 message_page
|
||||||
|
|
||||||
|
This method should be used whenever a screen that displays a simple message
|
||||||
|
is required.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub message_page
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my %params = @_;
|
||||||
|
|
||||||
|
my $title = $params{title};
|
||||||
|
my $message_box = $params{text};
|
||||||
|
|
||||||
|
my $left = defined $params{left} ? $params{left} : gettext("Back");
|
||||||
|
my $right = defined $params{right} ? $params{right} : gettext("Next");
|
||||||
|
|
||||||
|
$self->screen ("--title", $title,
|
||||||
|
"--cancel-label", $left,
|
||||||
|
"--ok-label", $right,
|
||||||
|
"--clear",
|
||||||
|
"--msgbox", $message_box,
|
||||||
|
SCREEN_ROWS,
|
||||||
|
SCREEN_COLUMNS,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 tryagain_page
|
||||||
|
|
||||||
|
This method displays a simple "try again" screen.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub tryagain_page
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my %params = @_;
|
||||||
|
|
||||||
|
my $title = $params{title};
|
||||||
|
my $choice = $params{choice};
|
||||||
|
|
||||||
|
my $try_again = "; " . gettext("please try again");
|
||||||
|
|
||||||
|
my $message_box = $title . ":'${choice}'" . $try_again;
|
||||||
|
|
||||||
|
$self->screen ("--title", $title,
|
||||||
|
"--cancel-label", gettext("Back"),
|
||||||
|
"--ok-label", gettext("Next"),
|
||||||
|
"--clear",
|
||||||
|
"--msgbox", $message_box,
|
||||||
|
SCREEN_ROWS,
|
||||||
|
SCREEN_COLUMNS,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 password_page
|
||||||
|
|
||||||
|
This method displays a screen suitable for entering a password.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub password_page
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my %params = @_;
|
||||||
|
|
||||||
|
my $title = $params{title};
|
||||||
|
my $message_box = $params{text};
|
||||||
|
|
||||||
|
my $left = defined $params{left} ? $params{left} : gettext("Back");
|
||||||
|
my $right = defined $params{right} ? $params{right} : gettext("Next");
|
||||||
|
|
||||||
|
$self->dialog ("--title", $title,
|
||||||
|
"--insecure",
|
||||||
|
"--cancel-label", $left,
|
||||||
|
"--ok-label", $right,
|
||||||
|
"--clear",
|
||||||
|
"--passwordbox", "\n" . $message_box,
|
||||||
|
SCREEN_ROWS,
|
||||||
|
SCREEN_COLUMNS,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 yesno_page
|
||||||
|
|
||||||
|
This method displays a simple yes/no screen, so the user can make a
|
||||||
|
simple binary selection.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub yesno_page
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my %params = @_;
|
||||||
|
|
||||||
|
my $title = $params{title};
|
||||||
|
my $text = $params{text};
|
||||||
|
|
||||||
|
my $left = defined $params{left} ? $params{left} : gettext("Yes");
|
||||||
|
my $right = defined $params{right} ? $params{right} : gettext("No");
|
||||||
|
my @args = (
|
||||||
|
"--title" => $title,
|
||||||
|
"--yes-label" => $left,
|
||||||
|
"--no-label" => $right,
|
||||||
|
);
|
||||||
|
push @args, "--defaultno" if defined $params{defaultno};
|
||||||
|
push @args, "--clear";
|
||||||
|
|
||||||
|
$self->screen (@args,
|
||||||
|
"--yesno", $text,
|
||||||
|
SCREEN_ROWS,
|
||||||
|
SCREEN_COLUMNS,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 input_page
|
||||||
|
|
||||||
|
This method displays a simple input screen with an input box.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub input_page
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my %params = @_;
|
||||||
|
|
||||||
|
my $title = $params{title};
|
||||||
|
my $text = $params{text};
|
||||||
|
my $value = $params{value};
|
||||||
|
|
||||||
|
my $left = defined $params{left} ? $params{left} : gettext("Back");
|
||||||
|
my $right = defined $params{right} ? $params{right} : gettext("Next");
|
||||||
|
|
||||||
|
$self->screen("--title", $title,
|
||||||
|
"--cancel-label", $left,
|
||||||
|
"--ok-label", $right,
|
||||||
|
"--clear",
|
||||||
|
"--inputbox", $text,
|
||||||
|
SCREEN_ROWS,
|
||||||
|
SCREEN_COLUMNS,
|
||||||
|
$value
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 infobox
|
||||||
|
|
||||||
|
This method is similar to a messagebox, but exits immediately, without clearing the screen.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub infobox
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my %params = @_;
|
||||||
|
my $title = $params{title};
|
||||||
|
my $text = $params{text};
|
||||||
|
my $height = $params{height} || "8";
|
||||||
|
my $width = $params{width} || SCREEN_COLUMNS;
|
||||||
|
$self->screen("--title", $title,
|
||||||
|
"--infobox", $text,
|
||||||
|
$height,
|
||||||
|
$width,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=head2 textbox
|
||||||
|
|
||||||
|
A text box lets you display the contents of a text file in a dialog box.
|
||||||
|
It is like a simple text file viewer.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub textbox
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my %params = @_;
|
||||||
|
my $title = $params{title};
|
||||||
|
my $file = $params{file};
|
||||||
|
my $height = $params{height} || '20';
|
||||||
|
my $width = $params{width} || SCREEN_COLUMNS;
|
||||||
|
$self->screen("--title", $title,
|
||||||
|
"--textbox", $file,
|
||||||
|
$height,
|
||||||
|
$width,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 menu_page
|
||||||
|
|
||||||
|
This method displays a screen with a menu.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub menu_page
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my %params = @_;
|
||||||
|
|
||||||
|
my $title = $params{title};
|
||||||
|
my $text = $params{text};
|
||||||
|
my @args = ("--clear", "--title", $title);
|
||||||
|
if ($params{default})
|
||||||
|
{
|
||||||
|
push @args, "--default-item", $params{default};
|
||||||
|
}
|
||||||
|
|
||||||
|
my $value = $params{value};
|
||||||
|
my $argsref = $params{argsref};
|
||||||
|
|
||||||
|
my $menu_rows = scalar @$argsref / 2;
|
||||||
|
|
||||||
|
$menu_rows = 10 if ($menu_rows > 10);
|
||||||
|
|
||||||
|
my $left = defined $params{left} ? $params{left} : gettext("Back");
|
||||||
|
my $right = defined $params{right} ? $params{right} : gettext("Next");
|
||||||
|
|
||||||
|
$self->dialog(@args,
|
||||||
|
"--cancel-label", $left,
|
||||||
|
"--ok-label", $right,
|
||||||
|
"--menu", $text,
|
||||||
|
SCREEN_ROWS,
|
||||||
|
SCREEN_COLUMNS,
|
||||||
|
$menu_rows,
|
||||||
|
@$argsref,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 keep_option
|
||||||
|
|
||||||
|
??
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub keep_option
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my ($value) = @_;
|
||||||
|
|
||||||
|
my $keep_phrase = gettext("Keep the current setting");
|
||||||
|
|
||||||
|
return ( gettext("keep"), "${keep_phrase}: $value" );
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 gauge
|
||||||
|
|
||||||
|
This method displays a progress bar. It takes a coderef as parameter, and uses
|
||||||
|
the coderef to drive the --gauge widget of the dialog program, as well as to
|
||||||
|
perform whatever actions are being reported by the progress bar. The coderef
|
||||||
|
should take one parameter, which is the file handle to write the controlling
|
||||||
|
text to. If the return value of the coderef is defined, it is displayed by a
|
||||||
|
message_page after the progress bar terminates.
|
||||||
|
|
||||||
|
All text used to update the progress bar should either be numbers between 0
|
||||||
|
and 100, or arbitrary text sandwiched between leading and training lines
|
||||||
|
of 'XXX' followed by newline. The numbers will update the percentage complete
|
||||||
|
of the display, and the text will update the displayed text. Updating the
|
||||||
|
displayed text will reset the precentage complete to 0, so text should always
|
||||||
|
be followed by number.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub gauge
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $sub = shift;
|
||||||
|
my %params = @_;
|
||||||
|
my $title = $params{title} || 'Progress';
|
||||||
|
my $feedback_title = $params{feedback_title} || 'Status';
|
||||||
|
my $init_text = $params{text} || 'Progress';
|
||||||
|
my @args = (
|
||||||
|
'--backtitle', $self->backtitle,
|
||||||
|
'--title', gettext($title),
|
||||||
|
);
|
||||||
|
push @args, "--clear" if $params{clear};
|
||||||
|
push @args, "--colors" if $params{colors};
|
||||||
|
push @args, "--no-collapse" if $params{no_collapse};
|
||||||
|
|
||||||
|
use FileHandle;
|
||||||
|
|
||||||
|
unless (open(WR, '|-'))
|
||||||
|
{
|
||||||
|
exec('/usr/bin/dialog',
|
||||||
|
@args,
|
||||||
|
'--gauge',
|
||||||
|
gettext($init_text),
|
||||||
|
SCREEN_ROWS,
|
||||||
|
SCREEN_COLUMNS,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
WR->autoflush(1);
|
||||||
|
my $text = &$sub(*WR);
|
||||||
|
close(WR);
|
||||||
|
$self->message_page('title' => $feedback_title, 'text' => $text)
|
||||||
|
if defined $text;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 run_screens
|
||||||
|
|
||||||
|
This method takes a directory of screens to run, and runs them in order.
|
||||||
|
To support navigation between screens, this method respects an integer
|
||||||
|
return value from the screens.
|
||||||
|
|
||||||
|
0 = all is well, continue to the next screen
|
||||||
|
1 = all is not well, go back to the previous screen
|
||||||
|
2 = catastrophic failure - return from run_screen
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub run_screens
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my ($subdir) = @_;
|
||||||
|
|
||||||
|
my $dir = CONSOLE_SCREENS . "/$subdir";
|
||||||
|
|
||||||
|
# This is fine. Noop if the directory isn't there.
|
||||||
|
unless (-d $dir)
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
# This is not fine. If it's there, we should be able to open it.
|
||||||
|
unless ( opendir(SCREENS, $dir) )
|
||||||
|
{
|
||||||
|
warn "Failed to open directory $dir: $!\n";
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
my @screens = sort grep (!/^(\.\.?)$/, readdir (SCREENS));
|
||||||
|
|
||||||
|
my @previous_screens = ();
|
||||||
|
while (@screens)
|
||||||
|
{
|
||||||
|
my $screen = shift @screens;
|
||||||
|
unless ( $screen =~ /(S\d\d[\d\w]+)/ )
|
||||||
|
{
|
||||||
|
warn "Unknown screen type $dir/$screen\n";
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
$screen = $1;
|
||||||
|
my $rv = system( "$dir/$screen" );
|
||||||
|
$rv >>= 8;
|
||||||
|
if ($rv == 0)
|
||||||
|
{
|
||||||
|
# Success, move to next screen.
|
||||||
|
push @previous_screens, $screen;
|
||||||
|
}
|
||||||
|
elsif ($rv == 1)
|
||||||
|
{
|
||||||
|
# Failure, go back one screen.
|
||||||
|
unshift @screens, $screen;
|
||||||
|
if (@previous_screens)
|
||||||
|
{
|
||||||
|
unshift @screens, pop @previous_screens;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
# We're at the beginning of the stack. Just return.
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
# Catastrophic failure, return. While 2 is the agreed-upon
|
||||||
|
# return code for this, consider it a catastrophic failure
|
||||||
|
# if we don't get a valid return code.
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
SME Server Developers <smebugs@mitel.com>
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
743
root/usr/share/perl5/vendor_perl/esmith/db.pm
Normal file
743
root/usr/share/perl5/vendor_perl/esmith/db.pm
Normal file
@ -0,0 +1,743 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::db;
|
||||||
|
|
||||||
|
use esmith::config;
|
||||||
|
use strict;
|
||||||
|
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::db - Routines for handling the e-smith configuration database
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
B<THIS MODULE HAS BEEN DEPRECATED>
|
||||||
|
|
||||||
|
use esmith::db;
|
||||||
|
use esmith::config;
|
||||||
|
|
||||||
|
my %config;
|
||||||
|
tie %config, 'esmith::config', $config_file;
|
||||||
|
|
||||||
|
db_set(\%config, $key, $type, \%properties);
|
||||||
|
db_set_type(\%config, $key, $type);
|
||||||
|
db_set_prop(\%config, $key, $property => $new_value);
|
||||||
|
|
||||||
|
my($type, %properties) = db_get(\%config, $key);
|
||||||
|
my $type = db_get_type(\%config, $key);
|
||||||
|
my %properties = db_get_prop(\%config, $key);
|
||||||
|
my $value = db_get_prop(\%config, $key, $property);
|
||||||
|
|
||||||
|
db_delete(\%config, $key);
|
||||||
|
db_delete_prop(\%config, $key, $property);
|
||||||
|
|
||||||
|
db_print(\%config, $key);
|
||||||
|
db_show(\%config, $key);
|
||||||
|
db_print_type(\%config, $key);
|
||||||
|
db_print_prop(\%config, $key, $prop);
|
||||||
|
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
B<THIS MODULE HAS BEEN DEPRECATED>. Please use a subclass of
|
||||||
|
esmith::DB::db instead, such as esmith::AccountsDB or esmith::ConfigDB.
|
||||||
|
|
||||||
|
I<Do not try to change this module>. Much code depends on subtle
|
||||||
|
nuances and bugs and you will break things if you try to fix it.
|
||||||
|
Instead, move any existing code away from esmith::db and towards
|
||||||
|
esmith::DB::db.
|
||||||
|
|
||||||
|
|
||||||
|
This module provides utility routines for manipulating e-smith
|
||||||
|
configuration data. OO and non-OO versions of the routines are provided.
|
||||||
|
For example, db_set() is the non-OO while set() can be called with an
|
||||||
|
object reference.
|
||||||
|
|
||||||
|
E-Smith DB entries have three parts. A key, a type and a hash of
|
||||||
|
properties.
|
||||||
|
|
||||||
|
key squid
|
||||||
|
type cephalopod
|
||||||
|
properties arms => 10
|
||||||
|
species => Loligo
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
use vars qw($VERSION @ISA @EXPORT);
|
||||||
|
$VERSION = sprintf "%d.%03d", q$Revision: 1.16 $ =~ /(\d+)\.(\d+)/;
|
||||||
|
|
||||||
|
use Exporter;
|
||||||
|
@ISA = qw(Exporter);
|
||||||
|
@EXPORT = qw(
|
||||||
|
db_set
|
||||||
|
db_get
|
||||||
|
db_delete
|
||||||
|
|
||||||
|
db_set_type
|
||||||
|
db_get_type
|
||||||
|
|
||||||
|
db_get_prop
|
||||||
|
db_set_prop
|
||||||
|
db_delete_prop
|
||||||
|
|
||||||
|
db_print
|
||||||
|
db_show
|
||||||
|
|
||||||
|
db_print_type
|
||||||
|
db_print_prop
|
||||||
|
);
|
||||||
|
|
||||||
|
=head2 Functions
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item B<db_set>
|
||||||
|
|
||||||
|
my $success = db_set(\%config, $key, $raw_value);
|
||||||
|
my $success = db_set(\%config, $key, $type);
|
||||||
|
my $success = db_set(\%config, $key, $type, \%properties);
|
||||||
|
|
||||||
|
Enters a new $key into the %config or replaces an existing $key. It
|
||||||
|
sets the $type and optionally %properties.
|
||||||
|
|
||||||
|
As a "bug which has become a feature" you can feed db_set() the
|
||||||
|
$raw_value for a $key (ie. 'type|prop1|val1|prop2|val2') and it will
|
||||||
|
setup the types and properties properly. I<Do not depend on this> but
|
||||||
|
don't change it either. There's code that depends on this behavior.
|
||||||
|
|
||||||
|
It returns true on success, false on failure.
|
||||||
|
|
||||||
|
If the $key contains a newline it will fail.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_set
|
||||||
|
{
|
||||||
|
my ($hash, $key, $new_value, $hashref) = @_;
|
||||||
|
|
||||||
|
return undef if ($key =~ /\n/);
|
||||||
|
if (defined $hashref)
|
||||||
|
{
|
||||||
|
my $properties = _db_hash_to_string($hashref);
|
||||||
|
if (defined $properties && $properties ne '')
|
||||||
|
{
|
||||||
|
$new_value .= "|$properties";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$new_value and $new_value =~ s/\n/\\n/g;
|
||||||
|
$$hash{$key} = $new_value;
|
||||||
|
return undef unless defined db_get($hash, $key);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 B<db_get>
|
||||||
|
|
||||||
|
my($type, %properties) = db_get(\%config, $key);
|
||||||
|
my $raw_value = db_get(\%config, $key);
|
||||||
|
my @keys = db_get(\%config);
|
||||||
|
|
||||||
|
Reads the $type and %properties for the given $key in %config.
|
||||||
|
|
||||||
|
In scalar context it returns the raw value of $config{$key} B<NOT> the
|
||||||
|
type! But it unescapes newlines. I<Use db_get_type() instead>.
|
||||||
|
|
||||||
|
If a $key is not given it returns all the @keys in the %config.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_get
|
||||||
|
{
|
||||||
|
my ($hash, $key) = @_;
|
||||||
|
|
||||||
|
return sort keys %$hash unless defined $key;
|
||||||
|
return undef unless exists $$hash{$key};
|
||||||
|
|
||||||
|
my $value = $$hash{$key};
|
||||||
|
$value and $value =~ s/\\n/\n/g;
|
||||||
|
return wantarray() ? _db_string_to_type_and_hash($value) : $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<db_delete>
|
||||||
|
|
||||||
|
db_delete(\%config, $key)
|
||||||
|
|
||||||
|
Deletes the $key from %config.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_delete
|
||||||
|
{
|
||||||
|
my ($hash, $key) = @_;
|
||||||
|
|
||||||
|
return undef unless defined db_get($hash, $key);
|
||||||
|
|
||||||
|
delete $$hash{$key};
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<db_set_type>
|
||||||
|
|
||||||
|
my $success = db_set_type(\%config, $key, $type)
|
||||||
|
|
||||||
|
Sets the $type for $config{$key}.
|
||||||
|
|
||||||
|
Returns true if the set succeeded, false otherwise.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_set_type
|
||||||
|
{
|
||||||
|
my ($hash, $key, $type) = @_;
|
||||||
|
|
||||||
|
return undef unless defined db_get($hash, $key);
|
||||||
|
|
||||||
|
my %properties = db_get_prop($hash, $key);
|
||||||
|
|
||||||
|
return db_set($hash, $key, $type, \%properties);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<db_get_type>
|
||||||
|
|
||||||
|
my $type = db_get_type(\%config, $key);
|
||||||
|
|
||||||
|
Returns the $type associated with the $key in the %config database.
|
||||||
|
|
||||||
|
Will return undef if the $key doesn't exist.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_get_type
|
||||||
|
{
|
||||||
|
my ($hash, $key) = @_;
|
||||||
|
|
||||||
|
return undef unless defined db_get($hash, $key);
|
||||||
|
|
||||||
|
my ($type) =
|
||||||
|
_db_string_to_type_and_hash(db_get($hash, $key));
|
||||||
|
return $type;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<db_set_prop>
|
||||||
|
|
||||||
|
my $success = db_set_prop(\%config, $key, $property => $new_value)
|
||||||
|
|
||||||
|
Sets the given $property of the $key in the %config database to the
|
||||||
|
$new_value. If the $property didn't exist, it will be added.
|
||||||
|
|
||||||
|
Returns true/value if it succeeded/failed.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_set_prop
|
||||||
|
{
|
||||||
|
my ($hash, $key, $prop, $new_value) = @_;
|
||||||
|
|
||||||
|
return undef unless defined db_get($hash, $key);
|
||||||
|
|
||||||
|
my $type = db_get_type($hash, $key);
|
||||||
|
my %properties = db_get_prop($hash, $key);
|
||||||
|
$properties{$prop} = $new_value;
|
||||||
|
return db_set($hash, $key, $type, \%properties);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<db_get_prop>
|
||||||
|
|
||||||
|
my %properties = db_get_prop(\%config, $key);
|
||||||
|
my $value = db_get_prop(\%config, $key, $property);
|
||||||
|
|
||||||
|
Returns the %properties for a $key in the %config database. If you
|
||||||
|
ask for a specific $property you'll get the $value for that $property.
|
||||||
|
|
||||||
|
Returns undef if the $key or $property doesn't exist.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_get_prop
|
||||||
|
{
|
||||||
|
my ($hash, $key, $prop) = @_;
|
||||||
|
|
||||||
|
my $val = db_get($hash, $key);
|
||||||
|
return (defined $prop ? undef : ()) unless defined $val;
|
||||||
|
|
||||||
|
my($type, %properties) = _db_string_to_type_and_hash($val);
|
||||||
|
|
||||||
|
return %properties unless defined $prop;
|
||||||
|
return undef unless exists $properties{$prop};
|
||||||
|
return $properties{$prop};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<db_delete_prop>
|
||||||
|
|
||||||
|
db_delete_prop(\%config, $key, $property)
|
||||||
|
|
||||||
|
Deletes a $property from the $key in the %config.
|
||||||
|
|
||||||
|
Returns undef if the $key doesn't exist.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_delete_prop
|
||||||
|
{
|
||||||
|
my ($hash, $key, $prop) = @_;
|
||||||
|
|
||||||
|
return undef unless defined db_get($hash, $key);
|
||||||
|
|
||||||
|
my $type = db_get_type($hash, $key);
|
||||||
|
my %properties = db_get_prop($hash, $key);
|
||||||
|
delete $properties{$prop};
|
||||||
|
return db_set($hash, $key, $type, \%properties);
|
||||||
|
}
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head2 Debugging Functions
|
||||||
|
|
||||||
|
These functions are useful for debugging.
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item B<db_print>
|
||||||
|
|
||||||
|
db_print(\%config);
|
||||||
|
db_print(\%config, $key);
|
||||||
|
|
||||||
|
Prints out keys and raw values in the %config database. If $key is
|
||||||
|
given it prints the $key and its raw value. If no $key is given it
|
||||||
|
prints out all the keys and their raw values.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_print
|
||||||
|
{
|
||||||
|
my ($hash, $key) = @_;
|
||||||
|
|
||||||
|
my @list;
|
||||||
|
|
||||||
|
if (defined $key)
|
||||||
|
{
|
||||||
|
return undef unless defined db_get($hash, $key);
|
||||||
|
@list = ($key);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
@list = db_get($hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
return undef unless scalar @list;
|
||||||
|
|
||||||
|
foreach (@list)
|
||||||
|
{
|
||||||
|
print "$_=", scalar db_get($hash, $_),"\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<db_show>
|
||||||
|
|
||||||
|
db_show(\%config);
|
||||||
|
db_show(\%config, $key);
|
||||||
|
|
||||||
|
Prints out keys and their values in a human readable format.
|
||||||
|
|
||||||
|
If $key is given it prints out the $key, type and properties of that
|
||||||
|
$key. Otherwise it prints out the key, type and properties for all
|
||||||
|
keys.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_show
|
||||||
|
{
|
||||||
|
my ($hash, $key) = @_;
|
||||||
|
|
||||||
|
my @list;
|
||||||
|
|
||||||
|
if (defined $key)
|
||||||
|
{
|
||||||
|
return undef unless defined db_get($hash, $key);
|
||||||
|
@list = ($key);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
@list = db_get($hash) unless defined $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
return undef unless scalar @list;
|
||||||
|
|
||||||
|
foreach (@list)
|
||||||
|
{
|
||||||
|
print "$_=";
|
||||||
|
|
||||||
|
my $type = db_get_type($hash, $_);
|
||||||
|
|
||||||
|
if (defined $type)
|
||||||
|
{
|
||||||
|
print "$type\n";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
print "\n";
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
my %properties = db_get_prop($hash, $_);
|
||||||
|
next unless scalar keys %properties;
|
||||||
|
|
||||||
|
foreach my $property (sort keys %properties)
|
||||||
|
{
|
||||||
|
print " $property=$properties{$property}\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<db_print_type>
|
||||||
|
|
||||||
|
db_print_type(\%config);
|
||||||
|
db_print_type(\%config, $key);
|
||||||
|
|
||||||
|
Prints out keys and their types in the %config database.
|
||||||
|
|
||||||
|
If $key is given, it prints out just that $key and its type.
|
||||||
|
Otherwise it prints out all the keys and their types.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_print_type
|
||||||
|
{
|
||||||
|
my ($hash, $key) = @_;
|
||||||
|
|
||||||
|
my @list;
|
||||||
|
|
||||||
|
if (defined $key)
|
||||||
|
{
|
||||||
|
return undef unless defined db_get($hash, $key);
|
||||||
|
@list = $key;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
@list = db_get($hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
return undef unless scalar @list;
|
||||||
|
|
||||||
|
foreach (@list)
|
||||||
|
{
|
||||||
|
print "$_=";
|
||||||
|
|
||||||
|
my $type = db_get_type($hash, $_);
|
||||||
|
|
||||||
|
print db_get_type($hash, $_),"\n" if defined $type;
|
||||||
|
print "\n" unless defined $type;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<db_print_prop>
|
||||||
|
|
||||||
|
db_print_prop(\%config, $key);
|
||||||
|
db_print_prop(\%config, $key, $property);
|
||||||
|
|
||||||
|
Prints out the properties (or a single $property) of the given $key in
|
||||||
|
the %config.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub db_print_prop
|
||||||
|
{
|
||||||
|
my ($hash, $key, $prop) = @_;
|
||||||
|
|
||||||
|
my @list;
|
||||||
|
my %list;
|
||||||
|
|
||||||
|
return undef unless defined db_get($hash, $key);
|
||||||
|
|
||||||
|
if (defined $prop)
|
||||||
|
{
|
||||||
|
my $value = db_get_prop($hash, $key, $prop);
|
||||||
|
return undef unless defined $value;
|
||||||
|
|
||||||
|
%list = ($prop => $value);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
%list = db_get_prop($hash, $key);
|
||||||
|
}
|
||||||
|
|
||||||
|
return undef unless scalar keys %list;
|
||||||
|
|
||||||
|
foreach (sort keys %list)
|
||||||
|
{
|
||||||
|
print "$_=$list{$_}\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=head2 OO Interface
|
||||||
|
|
||||||
|
To add to the confusion, esmith::db has a vestigal object-oriented
|
||||||
|
interface. Use esmith::DB::db instead.
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item B<new>
|
||||||
|
|
||||||
|
my $db = esmith::db->new($db_file)
|
||||||
|
|
||||||
|
Generates a new esmith::db object from the given $db_file
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub new
|
||||||
|
{
|
||||||
|
my($class,$dbname) = @_;
|
||||||
|
|
||||||
|
return $class->open($dbname);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<open>
|
||||||
|
|
||||||
|
my $db = esmith::db->open($db_name);
|
||||||
|
my $db = esmith::db->open($db_file);
|
||||||
|
|
||||||
|
Takes a database name (or pathname) and opens the named database.
|
||||||
|
The database name form is preferred over the explicit pathname.
|
||||||
|
|
||||||
|
For example
|
||||||
|
|
||||||
|
$db->open( 'configuration' );
|
||||||
|
or
|
||||||
|
$db->open( '/path/to/configuration' );
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub open
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $dbName = shift;
|
||||||
|
my $class = ref($self) || $self;
|
||||||
|
|
||||||
|
my $dataFile = _db_path($dbName);
|
||||||
|
|
||||||
|
unless ( $esmith::db::REFCOUNT{$dataFile} )
|
||||||
|
{
|
||||||
|
warn "Reading $dataFile into cache\n" if ($esmith::db::DEBUG);
|
||||||
|
|
||||||
|
my %db;
|
||||||
|
tie %db, 'esmith::config', $dataFile;
|
||||||
|
|
||||||
|
$esmith::db::CACHE{$dataFile} = \%db;
|
||||||
|
}
|
||||||
|
|
||||||
|
$self = bless {
|
||||||
|
DBNAME => $dataFile,
|
||||||
|
CACHE => $esmith::db::CACHE{$dataFile}
|
||||||
|
}, $class;
|
||||||
|
|
||||||
|
$esmith::db::REFCOUNT{$dataFile}++;
|
||||||
|
|
||||||
|
return $self;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub DESTROY
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
return $self->close();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<close>
|
||||||
|
|
||||||
|
$db->close;
|
||||||
|
|
||||||
|
Closes this database.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub close
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
|
||||||
|
my $dataFile = $self->{'DBNAME'};
|
||||||
|
|
||||||
|
$esmith::db::REFCOUNT{$dataFile}--;
|
||||||
|
|
||||||
|
if ( $esmith::db::REFCOUNT{$dataFile} == 0 )
|
||||||
|
{
|
||||||
|
delete $esmith::db::CACHE{$dataFile};
|
||||||
|
warn "esmith::db::close Closing $dataFile\n" if ($esmith::db::DEBUG);
|
||||||
|
}
|
||||||
|
elsif ( $esmith::db::REFCOUNT{$dataFile} > 0 )
|
||||||
|
{
|
||||||
|
warn "esmith::db::close Not closing $dataFile, references ",
|
||||||
|
$esmith::db::REFCOUNT{$dataFile}, "\n" if ($esmith::db::DEBUG);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$esmith::db::REFCOUNT{$dataFile} = 0;
|
||||||
|
warn "esmith::db::close Not closing $dataFile, zero references\n"
|
||||||
|
if ($esmith::db::DEBUG);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=item B<set>
|
||||||
|
|
||||||
|
=item B<set_type>
|
||||||
|
|
||||||
|
=item B<set_prop>
|
||||||
|
|
||||||
|
=item B<get>
|
||||||
|
|
||||||
|
=item B<get_type>
|
||||||
|
|
||||||
|
=item B<get_prop>
|
||||||
|
|
||||||
|
=item B<delete>
|
||||||
|
|
||||||
|
=item B<delete_prop>
|
||||||
|
|
||||||
|
These all work like their functional versions (ie. set() == db_set())
|
||||||
|
except it is not necessary to input the %config database.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub AUTOLOAD
|
||||||
|
{
|
||||||
|
no strict 'refs';
|
||||||
|
|
||||||
|
# fully qualified sub-name stored in $AUTOLOAD package variable
|
||||||
|
$esmith::db::AUTOLOAD =~ /^(.*::)(.*)$/;
|
||||||
|
my ($pkg, $sub) = ($1, $2);
|
||||||
|
|
||||||
|
# use *foo{THING} syntax to check if sub is defined (see perlref)
|
||||||
|
if (defined *{"${pkg}db_${sub}"}{CODE})
|
||||||
|
{
|
||||||
|
my $self = shift;
|
||||||
|
my $cache = $esmith::db::CACHE{$self->{DBNAME}};
|
||||||
|
wantarray ? return (my @p = &{"${pkg}db_${sub}"}($cache, @_))
|
||||||
|
: return (my $p = &{"${pkg}db_${sub}"}($cache, @_));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=begin private
|
||||||
|
|
||||||
|
=head2 Private functions
|
||||||
|
|
||||||
|
=over4
|
||||||
|
|
||||||
|
=item B<_db_hash_to_string>
|
||||||
|
|
||||||
|
my $raw_value = _db_hash_to_string($hashref);
|
||||||
|
|
||||||
|
Takes a reference to a hash and returns a string of pipe "|" delimited
|
||||||
|
pairs suitable for being stored.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _db_hash_to_string
|
||||||
|
{
|
||||||
|
my ($hash) = @_;
|
||||||
|
my $string = '';
|
||||||
|
|
||||||
|
foreach (sort keys %$hash)
|
||||||
|
{
|
||||||
|
$string .= '|' if length($string);
|
||||||
|
$string .= "$_|";
|
||||||
|
$string .= $$hash{$_} if defined $$hash{$_};
|
||||||
|
}
|
||||||
|
|
||||||
|
return $string;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=item B<_db_string_to_type_and_hash>
|
||||||
|
|
||||||
|
my($type, %properties) = _db_string_to_type_and_hash($raw_value);
|
||||||
|
|
||||||
|
Takes the $raw_value, which is a | delimited string, and spits it up
|
||||||
|
into the $type (the first field) and its %properties (the rest).
|
||||||
|
|
||||||
|
Escaped pipes (\|) are properly ignored as a delimiter.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _db_string_to_type_and_hash ($)
|
||||||
|
{
|
||||||
|
my ($arg) = @_;
|
||||||
|
return ('', ()) unless defined $arg;
|
||||||
|
|
||||||
|
# The funky regex is to avoid escaped pipes.
|
||||||
|
# If you specify a negative limit empty trailing fields are omitted.
|
||||||
|
return split(/(?<!\\)\|/, $arg, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<_db_path>
|
||||||
|
|
||||||
|
my $dfile = _db_path($database_name);
|
||||||
|
|
||||||
|
Takes a $database_name and returns the $file where it lives.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _db_path($)
|
||||||
|
{
|
||||||
|
my ($file) = @_;
|
||||||
|
|
||||||
|
if ($file =~ m:^/:)
|
||||||
|
{
|
||||||
|
return $file;
|
||||||
|
}
|
||||||
|
return "/home/e-smith/db/$file" if (-e "/home/e-smith/db/$file");
|
||||||
|
|
||||||
|
if (-e "/home/e-smith/$file")
|
||||||
|
{
|
||||||
|
warn "Database found in old location /home/e-smith/$file";
|
||||||
|
return "/home/e-smith/$file";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return "/home/e-smith/db/$file";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=end private
|
||||||
|
|
||||||
|
|
||||||
|
=head1 BUGS and CAVEATS
|
||||||
|
|
||||||
|
keys cannot contain newlines or pipes.
|
||||||
|
|
||||||
|
types and properties cannot contain pipes.
|
||||||
|
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
Mitel Networks Corporation
|
||||||
|
|
||||||
|
For more information, see http://www.e-smith.org/
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
133
root/usr/share/perl5/vendor_perl/esmith/ethernet.pm
Normal file
133
root/usr/share/perl5/vendor_perl/esmith/ethernet.pm
Normal file
@ -0,0 +1,133 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2005 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::ethernet;
|
||||||
|
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use File::Basename;
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::ethernet - Ethernet-related utility routines for e-smith
|
||||||
|
|
||||||
|
=head1 VERSION
|
||||||
|
|
||||||
|
This file documents C<esmith::ethernet> version B<1.4.0>
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::ethernet;
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module contains routines for
|
||||||
|
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head2 probeAdapters()
|
||||||
|
|
||||||
|
Probe for any recognised adapters
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub probeAdapters ()
|
||||||
|
{
|
||||||
|
opendir(my $dh, "/sys/class/net") or die "Couldn't open /sys/class/net: $!";
|
||||||
|
my @nics = grep { $_ !~ m/^\./ } readdir($dh);
|
||||||
|
closedir($dh);
|
||||||
|
my $adapters = '';
|
||||||
|
my $index = 1;
|
||||||
|
foreach my $nic (@nics){
|
||||||
|
# Untaint $nic and makes sure the name looks OK
|
||||||
|
next unless ($nic =~ m/^(\w+[\.:]?\d+)$/);
|
||||||
|
$nic = $1;
|
||||||
|
next if (
|
||||||
|
# skip loopback
|
||||||
|
$nic eq 'lo' ||
|
||||||
|
# skip non links
|
||||||
|
!-l "/sys/class/net/$nic" ||
|
||||||
|
# skip wireless nics
|
||||||
|
-d "/sys/class/net/$nic/wireless" ||
|
||||||
|
-l "/sys/class/net/$nic/phy80211" ||
|
||||||
|
# skip bridges
|
||||||
|
-d "/sys/class/net/$nic/bridge" ||
|
||||||
|
# skip vlans
|
||||||
|
-f "/proc/net/vlan/$nic" ||
|
||||||
|
# skip bonds
|
||||||
|
-d "/sys/class/net/$nic/bonding" ||
|
||||||
|
# skip tun/tap
|
||||||
|
-f "/sys/class/net/$nic/tun_flags" ||
|
||||||
|
# skip dummy
|
||||||
|
-d "/sys/devices/virtual/net/$nic"
|
||||||
|
);
|
||||||
|
# Now we should be left only wth ethernet adapters
|
||||||
|
open HW, "/sys/class/net/$nic/address";
|
||||||
|
my $mac = join("", <HW>);
|
||||||
|
close HW;
|
||||||
|
# Check MAC Addr and untaint it
|
||||||
|
next unless ($mac =~ m/^(([\da-f]{2}:){5}[\da-f]{2})$/i);
|
||||||
|
$mac = $1;
|
||||||
|
# If the device is a slave of a bridge, it's real MAC
|
||||||
|
# address can be found in /proc/net/bonding/bondX
|
||||||
|
if (-l "/sys/class/net/$nic/master"){
|
||||||
|
my $bond = basename (readlink "/sys/class/net/$nic/master");
|
||||||
|
local $/ = '';
|
||||||
|
open SLAVES, "/proc/net/bonding/$bond";
|
||||||
|
my @slaves = <SLAVES>;
|
||||||
|
close SLAVES;
|
||||||
|
my @slaveInfo = grep { /^Slave\ Interface:\ $nic/m } @slaves;
|
||||||
|
foreach (split /\n+/, (join "", @slaveInfo)){
|
||||||
|
$mac = $1 if (/^Permanent\ HW\ addr:\ (.*)$/);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
chomp($mac);
|
||||||
|
my $driver = basename (readlink "/sys/class/net/$nic/device/driver");
|
||||||
|
# Untaint driver name
|
||||||
|
next unless ($driver =~ m/^([\w\-]+)$/);
|
||||||
|
$driver = $1;
|
||||||
|
my $bus = basename (readlink "/sys/class/net/$nic/device/subsystem");
|
||||||
|
my $desc = $nic;
|
||||||
|
if ($bus eq 'pci'){
|
||||||
|
my $dev = basename (readlink "/sys/class/net/$nic/device");
|
||||||
|
# Untaint $dev
|
||||||
|
if ($dev =~ m/^(\d+:\d+:\d+\.\d+)$/){
|
||||||
|
$dev = $1;
|
||||||
|
$desc = `/sbin/lspci -s $dev`;
|
||||||
|
# Extract only description
|
||||||
|
$desc =~ m/^.*:.*:\s+(.*)\s*/;
|
||||||
|
$desc = $1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
elsif ($bus eq 'virtio'){
|
||||||
|
$desc = 'Virtio Network Device';
|
||||||
|
}
|
||||||
|
# TODO: we should also try to get the description of USB devices
|
||||||
|
$adapters .= "EthernetDriver" . $index++ . "\t" . $driver . "\t" .
|
||||||
|
$mac . "\t" . "\"$desc\"" . "\t" . $nic ."\n";
|
||||||
|
}
|
||||||
|
return $adapters;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Return one to make the import process return success.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
1;
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=AUTHOR
|
||||||
|
|
||||||
|
SME Server Developers <bugs@e-smith.com>
|
||||||
|
|
||||||
|
For more information see http://www.e-smith.org/
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
220
root/usr/share/perl5/vendor_perl/esmith/event.pm
Normal file
220
root/usr/share/perl5/vendor_perl/esmith/event.pm
Normal file
@ -0,0 +1,220 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2005 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::event;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use Exporter;
|
||||||
|
use Time::HiRes qw( usleep ualarm gettimeofday tv_interval );
|
||||||
|
use esmith::Logger;
|
||||||
|
use File::Basename;
|
||||||
|
use File::Temp qw/ :mktemp /;
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
use POSIX;
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::event - Routines for handling e-smith events
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::event;
|
||||||
|
|
||||||
|
my $exitcode = event_signal($event, @args);
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
our $VERSION = sprintf '%d.%03d', q$Revision: 1.16 $ =~ /: (\d+).(\d+)/;
|
||||||
|
our @ISA = qw(Exporter);
|
||||||
|
our @EXPORT = qw(event_signal);
|
||||||
|
|
||||||
|
our @EXPORT_OK = ();
|
||||||
|
our %EXPORT_TAGS = ();
|
||||||
|
our $return_value = undef;
|
||||||
|
|
||||||
|
tie *LOG, 'esmith::Logger', 'esmith::event';
|
||||||
|
|
||||||
|
sub event_signal
|
||||||
|
{
|
||||||
|
my ($event, @args) = @_;
|
||||||
|
if ($event eq "actions")
|
||||||
|
{
|
||||||
|
warn("'actions' is not a valid event name.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $events = "/etc/e-smith/events";
|
||||||
|
my $handlerDir = "$events/$event";
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# get event handler filenames
|
||||||
|
#------------------------------------------------------------
|
||||||
|
opendir (DIR, $handlerDir)
|
||||||
|
|| die "Can't open directory $handlerDir\n";
|
||||||
|
|
||||||
|
# Create a hash of handlers (ignore directories),
|
||||||
|
# with value of basename(handler)
|
||||||
|
my %handlers = ();
|
||||||
|
foreach (grep {! -d "$handlerDir/$_"} readdir (DIR))
|
||||||
|
{
|
||||||
|
$handlers{"$handlerDir/$_"} = $_;
|
||||||
|
}
|
||||||
|
|
||||||
|
closedir (DIR);
|
||||||
|
|
||||||
|
# Add generic handlers to list, if their metadata directories
|
||||||
|
# exist
|
||||||
|
$handlers{"$events/actions/generic_template_expand"} = "S05generic_template_expand"
|
||||||
|
if ( -d "$handlerDir/templates2expand");
|
||||||
|
$handlers{"$events/actions/adjust-services"} = "S90adjust-services"
|
||||||
|
if ( -d "$handlerDir/services2adjust");
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# Execute all handlers, sending any output to the system log.
|
||||||
|
#
|
||||||
|
# Event handlers are not supposed to generate error messages
|
||||||
|
# under normal conditions, so we do not provide a mechanism
|
||||||
|
# for event handlers to signal errors to the user. Errors can
|
||||||
|
# only be written to the log file.
|
||||||
|
#------------------------------------------------------------
|
||||||
|
print LOG "Processing event: $event @args";
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# Run handlers, logging all output.
|
||||||
|
#------------------------------------------------------------
|
||||||
|
|
||||||
|
# assume success
|
||||||
|
my $exitcode = 1;
|
||||||
|
|
||||||
|
foreach my $filename
|
||||||
|
(sort { $handlers{$a} cmp $handlers{$b} } keys %handlers)
|
||||||
|
{
|
||||||
|
my $handler = basename $filename;
|
||||||
|
my $startTime = [gettimeofday];
|
||||||
|
|
||||||
|
my $status = -1;
|
||||||
|
if (-x $filename)
|
||||||
|
{
|
||||||
|
print LOG "Running event handler: $filename";
|
||||||
|
|
||||||
|
unless (($status = _mysystem(\*LOG, $filename, $event, @args)) == 0)
|
||||||
|
{
|
||||||
|
# if any handler fails, the entire event fails
|
||||||
|
$exitcode = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
print LOG "Skipping non-executable event handler: $filename";
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $endTime = [gettimeofday];
|
||||||
|
my $elapsedTime = tv_interval($startTime, $endTime);
|
||||||
|
my $log = "$handler=action|Event|$event|Action|$handler";
|
||||||
|
$log .= "|Start|@$startTime|End|@$endTime|Elapsed|$elapsedTime";
|
||||||
|
$log .= "|Status|$status" if $status;
|
||||||
|
print LOG $log;
|
||||||
|
}
|
||||||
|
|
||||||
|
# Implement event queuing for clustered systems.
|
||||||
|
my $qfifo = "/var/spool/eventq";
|
||||||
|
return $exitcode unless (-e $qfifo);
|
||||||
|
|
||||||
|
# Ensure we aren't called by a cascaded event. We only need to
|
||||||
|
# queue the top-level of such a beast.
|
||||||
|
my $ppid = getppid();
|
||||||
|
open F, "/proc/$ppid/cmdline";
|
||||||
|
my $cmd = <F>;
|
||||||
|
close F;
|
||||||
|
|
||||||
|
unless($cmd =~ "/etc/e-smith/event")
|
||||||
|
{
|
||||||
|
my $fd = POSIX::open($qfifo, &POSIX::O_WRONLY) or return $exitcode;
|
||||||
|
my $argstr = join(" ",$event,@args);
|
||||||
|
$argstr .= "\n";
|
||||||
|
POSIX::write($fd, $argstr, length($argstr));
|
||||||
|
POSIX::close($fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $exitcode;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _mysystem
|
||||||
|
{
|
||||||
|
my ($logger, $filename, $event, @args) = @_;
|
||||||
|
|
||||||
|
my $pid = open(PIPE, "-|");
|
||||||
|
die "Failed to fork: $!\n" unless defined $pid;
|
||||||
|
|
||||||
|
if ($pid)
|
||||||
|
{
|
||||||
|
# Parent
|
||||||
|
while (my $line = <PIPE>)
|
||||||
|
{
|
||||||
|
print $logger $line;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
# Child
|
||||||
|
open(STDERR, ">&STDOUT");
|
||||||
|
exec($filename, $event, @args);
|
||||||
|
}
|
||||||
|
close(PIPE);
|
||||||
|
return $?;
|
||||||
|
}
|
||||||
|
|
||||||
|
#------------------------------------------------------------
|
||||||
|
# Attempt to eval perl handlers for efficiency - not currently used
|
||||||
|
# return 1 on success; 0 on error
|
||||||
|
#------------------------------------------------------------
|
||||||
|
sub _runHandler($)
|
||||||
|
{
|
||||||
|
my ($filename) = @_;
|
||||||
|
|
||||||
|
open(FILE, $filename) || die "Couldn't open $filename: $!";
|
||||||
|
my @lines = <FILE>;
|
||||||
|
close FILE;
|
||||||
|
|
||||||
|
my $string = "";
|
||||||
|
|
||||||
|
unless ( $lines[0] =~ /^#!.*perl/ )
|
||||||
|
{
|
||||||
|
# STDOUT and STDERR are both redirected going to LOG
|
||||||
|
return (system($filename, @ARGV) == 0) ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
map { $string .= $_ } @lines;
|
||||||
|
|
||||||
|
print "Eval of $filename...";
|
||||||
|
|
||||||
|
# Override 'exit' in symbol table for handlers
|
||||||
|
sub exit { die "$_[0]\n" };
|
||||||
|
*CORE::GLOBAL::exit = \&esmith::event::exit;
|
||||||
|
|
||||||
|
my $status = eval $string;
|
||||||
|
chomp $@;
|
||||||
|
|
||||||
|
# if $@ is defined, then die or exit was called - use that status
|
||||||
|
$status = $@ if defined $@;
|
||||||
|
|
||||||
|
# for all exit values except 0, assume failure
|
||||||
|
if ($@)
|
||||||
|
{
|
||||||
|
print "Eval of $filename failed: $status\n";
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
print "$status\n";
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
61
root/usr/share/perl5/vendor_perl/esmith/lockfile.pm
Normal file
61
root/usr/share/perl5/vendor_perl/esmith/lockfile.pm
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::lockfile;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use vars qw($VERSION @ISA @EXPORT_OK);
|
||||||
|
use Exporter;
|
||||||
|
use Fcntl ":flock";
|
||||||
|
use FileHandle;
|
||||||
|
|
||||||
|
@ISA = qw(Exporter);
|
||||||
|
@EXPORT_OK = qw(
|
||||||
|
LockFileOrReturn LockFileOrWait UnlockFile
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
sub LockFileOrReturn ($)
|
||||||
|
{
|
||||||
|
# Attempt to lock a file. If the lock fails, return immediately.
|
||||||
|
|
||||||
|
my $lock_file = shift;
|
||||||
|
|
||||||
|
my $FH = new FileHandle;
|
||||||
|
|
||||||
|
$FH->open(">> $lock_file")
|
||||||
|
or die "Cannot open lock file $lock_file for writing: $!.\n";
|
||||||
|
|
||||||
|
flock($FH, LOCK_EX | LOCK_NB) or return 0;
|
||||||
|
|
||||||
|
return $FH;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub LockFileOrWait ($)
|
||||||
|
{
|
||||||
|
# Attempt to lock a file. Wait until the file is available.
|
||||||
|
|
||||||
|
my $lock_file = shift;
|
||||||
|
|
||||||
|
my $FH = new FileHandle;
|
||||||
|
|
||||||
|
$FH->open(">> $lock_file")
|
||||||
|
or die "Cannot open lock file $lock_file for writing: $!.\n";
|
||||||
|
|
||||||
|
flock($FH, LOCK_EX) or return 0;
|
||||||
|
|
||||||
|
return $FH;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub UnlockFile ($)
|
||||||
|
{
|
||||||
|
my $FH = shift;
|
||||||
|
|
||||||
|
flock($FH, LOCK_UN);
|
||||||
|
$FH->close;
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
76
root/usr/share/perl5/vendor_perl/esmith/logrotate.pm
Normal file
76
root/usr/share/perl5/vendor_perl/esmith/logrotate.pm
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::event;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::logrotate - Routines for handling rotation of log files
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::logrotate;
|
||||||
|
|
||||||
|
NewLogFileSymlink($file);
|
||||||
|
MakeFilenameFromSymlink($file);
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
BEGIN
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
sub NewLogFileSymlink
|
||||||
|
{
|
||||||
|
my $file = shift;
|
||||||
|
unless (defined $file)
|
||||||
|
{
|
||||||
|
warn("newlogfilesymlink called with no argument");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $time = time();
|
||||||
|
|
||||||
|
if (-f "/var/log/${file}")
|
||||||
|
{
|
||||||
|
my ($sec,$min,$hour,$mday,$mon,$year) = localtime($time - 1);
|
||||||
|
my $target = sprintf("%s%04d%02d%02d%02d%02d%02d",
|
||||||
|
$file, $year+1900, $mon, $mday, $hour, $min, $sec);
|
||||||
|
move("/var/log/${file}", "/var/log/${target}") or
|
||||||
|
die "Could not move /var/log/${file} to /var/log/${target}";
|
||||||
|
}
|
||||||
|
|
||||||
|
my ($sec,$min,$hour,$mday,$mon,$year) = localtime($time);
|
||||||
|
my $target = sprintf("%s%04d%02d%02d%02d%02d%02d",
|
||||||
|
$file, $year+1900, $mon, $mday, $hour, $min, $sec);
|
||||||
|
unlink("/var/log/${file}") or
|
||||||
|
warn "Could not unlink /var/log/${file}";
|
||||||
|
symlink("/var/log/${target}", "/var/log/${file}") or
|
||||||
|
warn "Could not symlink /var/log/${target} to /var/log/${file}";
|
||||||
|
}
|
||||||
|
|
||||||
|
sub MakeFilenameFromSymlink
|
||||||
|
{
|
||||||
|
use File::Basename;
|
||||||
|
my $filename = shift;
|
||||||
|
|
||||||
|
return $filename unless (-l $filename);
|
||||||
|
my $link = readlink $filename;
|
||||||
|
my $directory = dirname($filename);
|
||||||
|
return "${directory}/${link}";
|
||||||
|
}
|
||||||
|
|
||||||
|
END
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
140
root/usr/share/perl5/vendor_perl/esmith/tcpsvd.pm
Normal file
140
root/usr/share/perl5/vendor_perl/esmith/tcpsvd.pm
Normal file
@ -0,0 +1,140 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 2005-2006 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::tcpsvd;
|
||||||
|
use strict;
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::tcpsvd - Manage tcpsvd "peers" directory
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::tcpsvd;
|
||||||
|
|
||||||
|
esmith::tcpsvd::configure_peers($service)
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This module provides utility functions for use with tcpsvd from Gerrit
|
||||||
|
Pape's ipsvd package - see http://smarden.org/ipsvd/.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
use esmith::NetworksDB;
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
use esmith::util;
|
||||||
|
use esmith::lockfile;
|
||||||
|
use Carp;
|
||||||
|
|
||||||
|
=head2 configure_peers($service [, $peers_directory] )
|
||||||
|
|
||||||
|
configure_peers() configures the "peers" direectory used by tcpsvd for
|
||||||
|
access control and environment maipulation. $service identifies the name
|
||||||
|
of the service managed by supervise or runit. The peers directory (as
|
||||||
|
specified by the optional $peers_directory argument, defaulting to
|
||||||
|
"/var/service/$service/peers") is expected to contain files "0" defining
|
||||||
|
access conditions for public (default) accesss, and "local", defining
|
||||||
|
access conditions for local access. configure_peers() creates a set of
|
||||||
|
symlinks so that tcpsvd uses "local" for all local network access to
|
||||||
|
the service.
|
||||||
|
|
||||||
|
See http://smarden.org/ipsvd/ipsvd-instruct.5.html for all details of
|
||||||
|
the contents of the peers directory.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub configure_peers
|
||||||
|
{
|
||||||
|
my $service = shift;
|
||||||
|
my $peers = shift || "/var/service/$service/peers";
|
||||||
|
|
||||||
|
unless (opendir(PEERS, $peers))
|
||||||
|
{
|
||||||
|
carp "Cannot read peers directory: $!";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $config = esmith::ConfigDB->open;
|
||||||
|
unless ($config)
|
||||||
|
{
|
||||||
|
carp "Could not open config db.";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$service = $config->get($service);
|
||||||
|
unless ($service)
|
||||||
|
{
|
||||||
|
carp "No service record for $service";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
my $access = $service->prop('access') || "localhost";
|
||||||
|
my $nets = esmith::NetworksDB->open;
|
||||||
|
unless ($nets)
|
||||||
|
{
|
||||||
|
carp "Could not open networks db.";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $gw = $config->get('GatewayIP');
|
||||||
|
|
||||||
|
# Make a list of local networks, in prefix format
|
||||||
|
my %nets = ($access eq "localhost") ? () :
|
||||||
|
map
|
||||||
|
{
|
||||||
|
$_ => 1,
|
||||||
|
}
|
||||||
|
map
|
||||||
|
{
|
||||||
|
esmith::util::computeAllLocalNetworkPrefixes($_->key, $_->prop('Mask'));
|
||||||
|
}
|
||||||
|
($nets->get_all_by_prop('type', 'network'));
|
||||||
|
|
||||||
|
$nets{'127.0.0.1'} = 1;
|
||||||
|
|
||||||
|
# Setup lock on peers directory while we manipulate it
|
||||||
|
my $lock = esmith::lockfile::LockFileOrWait("$peers/local");
|
||||||
|
|
||||||
|
# Now manage a set of symlinks to the "local" instructions file
|
||||||
|
foreach my $insfile (readdir (PEERS))
|
||||||
|
{
|
||||||
|
next unless -l "$peers/$insfile";
|
||||||
|
if (exists $nets{$insfile})
|
||||||
|
{
|
||||||
|
# Cross this one off the list so that we don't bother creating it
|
||||||
|
delete $nets{$insfile};
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
# We no longer need this entry
|
||||||
|
unlink "$peers/$insfile" or
|
||||||
|
warn "Could not delete access control file $peers/$insfile: $!\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
closedir(PEERS);
|
||||||
|
|
||||||
|
foreach my $insfile (keys %nets)
|
||||||
|
{
|
||||||
|
symlink "local", "$peers/$insfile" or
|
||||||
|
warn "Cannot add instructions file for $peers/$insfile: $!\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (defined $gw)
|
||||||
|
{
|
||||||
|
# We have a defined gateway address - make sure that the router doesn't have
|
||||||
|
# relay privileges
|
||||||
|
my $gw_ip = $gw->value;
|
||||||
|
unlink "$peers/$gw_ip";
|
||||||
|
symlink "0", "$peers/$gw_ip" or
|
||||||
|
warn "Cannot add instructions file for $peers/$gw_ip: $!\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
# Unlock peers directory
|
||||||
|
$lock && esmith::lockfile::UnlockFile($lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
1091
root/usr/share/perl5/vendor_perl/esmith/templates.pm
Normal file
1091
root/usr/share/perl5/vendor_perl/esmith/templates.pm
Normal file
File diff suppressed because it is too large
Load Diff
1413
root/usr/share/perl5/vendor_perl/esmith/util.pm
Normal file
1413
root/usr/share/perl5/vendor_perl/esmith/util.pm
Normal file
File diff suppressed because it is too large
Load Diff
201
root/usr/share/perl5/vendor_perl/esmith/util/link.pm
Normal file
201
root/usr/share/perl5/vendor_perl/esmith/util/link.pm
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::util::link;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use esmith::ConfigDB;
|
||||||
|
require Exporter;
|
||||||
|
|
||||||
|
our @ISA = qw(Exporter);
|
||||||
|
our @EXPORT_OK = qw(getExternalLink);
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::util::link - utilities for manipulating network links
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::util::link qw(getExternalLink);
|
||||||
|
|
||||||
|
# wait at most 60 seconds for the link to come up
|
||||||
|
my $timeout = 60;
|
||||||
|
|
||||||
|
# now just get the link
|
||||||
|
if (getExternalLink($timeout))
|
||||||
|
{
|
||||||
|
# the link is up
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
# the link didn't come up
|
||||||
|
}
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This is a collection of generally useful functions for manipulating network
|
||||||
|
links.
|
||||||
|
|
||||||
|
Functions are exported only on request.
|
||||||
|
|
||||||
|
=head2 Functions
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item I<getExternalLink($timeout)>
|
||||||
|
|
||||||
|
Bring up the external link if it is not already up, waiting at most $timeout
|
||||||
|
seconds. If a $timeout is not specified, it defaults to 300 (5 minutes) for
|
||||||
|
dialup connections. This function can be used for both dialup and dedicated
|
||||||
|
connections, but dedicated connections will always return 1 (true).
|
||||||
|
|
||||||
|
Returns 1 if the external link is already up, or if it comes up within the
|
||||||
|
specfied $timeout period.
|
||||||
|
|
||||||
|
Returns 0 if the external link does not come up within the specified $timeout
|
||||||
|
period.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub getExternalLink
|
||||||
|
{
|
||||||
|
my $timeout = shift;
|
||||||
|
|
||||||
|
my $configdb = esmith::ConfigDB->open;
|
||||||
|
my $rec = $configdb->get("AccessType");
|
||||||
|
my $accessType = $rec->value;
|
||||||
|
if ($accessType eq "dialup")
|
||||||
|
{
|
||||||
|
return _getDialupLink($timeout);
|
||||||
|
}
|
||||||
|
elsif ($accessType eq "dedicated")
|
||||||
|
{
|
||||||
|
# assume we are up
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
# unknown access type
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=begin _private
|
||||||
|
|
||||||
|
=item I<getDialupLink($timeout)>
|
||||||
|
|
||||||
|
Bring up the ppp0 link, waiting at most $timeout seconds.
|
||||||
|
|
||||||
|
Returns 1 if the link comes up within the timeout period.
|
||||||
|
|
||||||
|
Returns 0 if the link does not come up within the timeout period.
|
||||||
|
|
||||||
|
The default timeout is 300 seconds.
|
||||||
|
|
||||||
|
=end _private
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _getDialupLink
|
||||||
|
{
|
||||||
|
local $|=1;
|
||||||
|
my $timeout = shift(@_) || 300;
|
||||||
|
|
||||||
|
# check for existing ppp link
|
||||||
|
if (-f "/var/run/ppp0.pid")
|
||||||
|
{
|
||||||
|
# already up - return 1
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
# create a diald monitor channel
|
||||||
|
my $ctlFile = "/etc/diald/diald.ctl";
|
||||||
|
my $monFile = "/tmp/diald.monitor.$$";
|
||||||
|
system('/bin/mknod', $monFile, 'p') == 0
|
||||||
|
or die "Can't mknod $monFile: $!\n";
|
||||||
|
|
||||||
|
# open control channel to diald
|
||||||
|
open (CTL, ">$ctlFile") or die "Can't open $ctlFile: $!\n";
|
||||||
|
|
||||||
|
# set up a child process to monitor the channel
|
||||||
|
|
||||||
|
my $pid = fork;
|
||||||
|
die "Can't fork: $!" unless defined $pid;
|
||||||
|
if ($pid)
|
||||||
|
{
|
||||||
|
# parent
|
||||||
|
|
||||||
|
# if the pipe reader isn't up first, diald will bail, so we open
|
||||||
|
# another pipe writer just to wait for the pipe reader
|
||||||
|
open (MON_W, ">$monFile") or die "can't open $monFile: $!\n";
|
||||||
|
|
||||||
|
# begin monitoring diald status via monitor fifo
|
||||||
|
print CTL "monitor $monFile\n";
|
||||||
|
close CTL;
|
||||||
|
|
||||||
|
# ok, everything is up and ready - send USR1 to diald
|
||||||
|
open (PID, "</var/run/diald.pid")
|
||||||
|
or die "can't open diald pidfile: $!\n";
|
||||||
|
my $dialdPid = <PID>;
|
||||||
|
close PID;
|
||||||
|
kill 'USR1', $dialdPid;
|
||||||
|
|
||||||
|
# Wait for the child to exit, then check for link again
|
||||||
|
waitpid($pid, 0);
|
||||||
|
close MON_W;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
# child
|
||||||
|
|
||||||
|
open (MON, "<$monFile") or die "Can't open $monFile: $!\n";
|
||||||
|
|
||||||
|
# Parse the diald monitor stream for state information
|
||||||
|
my $state = "";
|
||||||
|
my $elapsed = 0;
|
||||||
|
while (<MON>)
|
||||||
|
{
|
||||||
|
# lucky us; diald sends a STATUS msg every second
|
||||||
|
if (/^STATUS/)
|
||||||
|
{
|
||||||
|
$elapsed++;
|
||||||
|
if ($elapsed >= $timeout)
|
||||||
|
{
|
||||||
|
# time is up - exit with failure code
|
||||||
|
exit 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
elsif (/^STATE/)
|
||||||
|
{
|
||||||
|
$state = $_;
|
||||||
|
}
|
||||||
|
elsif ($state eq 'UP')
|
||||||
|
{
|
||||||
|
# the link is up - exit with success code
|
||||||
|
exit 0;
|
||||||
|
}
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
close MON;
|
||||||
|
# end child
|
||||||
|
}
|
||||||
|
|
||||||
|
# parent (cont)
|
||||||
|
|
||||||
|
unlink $monFile;
|
||||||
|
|
||||||
|
if ($? == 0 || -f "/var/run/ppp0.pid")
|
||||||
|
{
|
||||||
|
# ok we're up - return 1 (true)
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
# out of time - return 0 (false)
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
312
root/usr/share/perl5/vendor_perl/esmith/util/network.pm
Normal file
312
root/usr/share/perl5/vendor_perl/esmith/util/network.pm
Normal file
@ -0,0 +1,312 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::util::network;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
|
||||||
|
require Exporter;
|
||||||
|
our @ISA = qw(Exporter);
|
||||||
|
our @EXPORT_OK = qw(isValidIP cleanIP isValidPort cmpIP);
|
||||||
|
our %EXPORT_TAGS = (
|
||||||
|
all => [qw(isValidIP cleanIP isValidPort cmpIP)]
|
||||||
|
);
|
||||||
|
|
||||||
|
use Net::IPv4Addr qw(:all);
|
||||||
|
use Mail::RFC822::Address;
|
||||||
|
use esmith::AccountsDB;
|
||||||
|
|
||||||
|
use constant TRUE => 1;
|
||||||
|
use constant FALSE => 0;
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
use_ok('esmith::util::network');
|
||||||
|
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::util::network - miscellaneous network utilities
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::util::network qw(the functions you want);
|
||||||
|
|
||||||
|
my $ip = cleanIP($orig_ip);
|
||||||
|
my $is_valid = isValidIP($ip);
|
||||||
|
my $is_valid = isValidPort($port);
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This is a collection of generally useful functions for working with IP
|
||||||
|
addresses.
|
||||||
|
|
||||||
|
Functions are exported only on request.
|
||||||
|
|
||||||
|
=head2 Functions
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item I<cleanIP>
|
||||||
|
|
||||||
|
my $ip = cleanIP($orig_ip);
|
||||||
|
|
||||||
|
If the $orig_ip is valid it will be cleaned up into a cannonical form,
|
||||||
|
stripping any padding zeros and such.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use esmith::util::network qw(cleanIP);
|
||||||
|
|
||||||
|
my %ips = (
|
||||||
|
'000.000.000.000' => '0.0.0.0',
|
||||||
|
'0.0.0.0' => '0.0.0.0',
|
||||||
|
'001.2.003.4' => '1.2.3.4',
|
||||||
|
'100.2.3.4' => '100.2.3.4',
|
||||||
|
'10.13.14.015' => '10.13.14.15',
|
||||||
|
'10.33.15.109' => '10.33.15.109',
|
||||||
|
'1.2.3.4.5' => ''
|
||||||
|
);
|
||||||
|
|
||||||
|
while( my($ip, $cleanip) = each %ips ) {
|
||||||
|
is( cleanIP($ip), $cleanip, "cleanIP($ip)" );
|
||||||
|
}
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub cleanIP {
|
||||||
|
my $ip = shift;
|
||||||
|
|
||||||
|
return '' unless isValidIP($ip);
|
||||||
|
$ip =~ s/\b0+(\d+)/$1/g;
|
||||||
|
|
||||||
|
return isValidIP($ip) ? $ip : '';
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<isValidIP>
|
||||||
|
|
||||||
|
my $is_valid = isValidIP($ip);
|
||||||
|
|
||||||
|
Returns the IP given if $ip is a properly formatted IP address, undef otherwise.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use esmith::util::network qw(isValidIP);
|
||||||
|
|
||||||
|
my @goodIPs = qw(1.2.3.4
|
||||||
|
0.0.0.0
|
||||||
|
255.255.255.255
|
||||||
|
001.002.003.004
|
||||||
|
1.32.123.213
|
||||||
|
192.168.0.3
|
||||||
|
02.19.090.19
|
||||||
|
);
|
||||||
|
foreach my $ip (@goodIPs) {
|
||||||
|
ok( isValidIP($ip), "valid $ip");
|
||||||
|
}
|
||||||
|
|
||||||
|
my @badIPs = qw(256.3.2.4
|
||||||
|
-1.39.9.23
|
||||||
|
0
|
||||||
|
1
|
||||||
|
255.255.255.255.0
|
||||||
|
239..19.23.12
|
||||||
|
1.2.3.4.
|
||||||
|
foo.bar.com
|
||||||
|
);
|
||||||
|
|
||||||
|
foreach my $ip (@badIPs) {
|
||||||
|
ok( !isValidIP($ip), "invalid $ip");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub isValidIP($)
|
||||||
|
{
|
||||||
|
my ($string) = @_;
|
||||||
|
return unless defined ipv4_chkip($string);
|
||||||
|
return $string eq ipv4_chkip($string);
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<isValidPort>
|
||||||
|
|
||||||
|
my $is_valid = isValidPort($port);
|
||||||
|
|
||||||
|
Returns true if $port is a properly formatted port, false otherwise.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
@badports = (98765434, -183, 0, 'bad port', 'a');
|
||||||
|
@goodports = (67, 23, 1, 54736);
|
||||||
|
|
||||||
|
foreach $port (@badports) {
|
||||||
|
isnt(esmith::util::network::isValidPort($port), 1);
|
||||||
|
}
|
||||||
|
foreach $port (@goodports) {
|
||||||
|
is(esmith::util::network::isValidPort($port), 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub isValidPort($)
|
||||||
|
{
|
||||||
|
my $port = shift;
|
||||||
|
|
||||||
|
return FALSE unless defined $port;
|
||||||
|
|
||||||
|
if (($port =~ /^\d+$/) &&
|
||||||
|
($port > 0) &&
|
||||||
|
($port < 65536))
|
||||||
|
{
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<cmpIP>
|
||||||
|
|
||||||
|
Performs a cmp operation on two IP addresses.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
$ip1 = '24.123.212.87';
|
||||||
|
$ip2 = '240.34.216.12';
|
||||||
|
|
||||||
|
is(esmith::util::network::cmpIP($ip1, $ip2), -1);
|
||||||
|
is(esmith::util::network::cmpIP($ip2, $ip1), 1);
|
||||||
|
is(esmith::util::network::cmpIP($ip1, $ip1), 0);
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub cmpIP($$)
|
||||||
|
{
|
||||||
|
my $ip1 = ipv4_chkip(shift);
|
||||||
|
my $ip2 = ipv4_chkip(shift);
|
||||||
|
|
||||||
|
die "The first argument is not a valid IP address.\n" if not $ip1;
|
||||||
|
die "The second argument is not a valid IP address.\n" if not $ip2;
|
||||||
|
|
||||||
|
my @ip1cmps = split /\./, $ip1;
|
||||||
|
my @ip2cmps = split /\./, $ip2;
|
||||||
|
|
||||||
|
while (@ip1cmps)
|
||||||
|
{
|
||||||
|
my $cmp1 = shift @ip1cmps;
|
||||||
|
my $cmp2 = shift @ip2cmps;
|
||||||
|
|
||||||
|
my $cmp = $cmp1 <=> $cmp2;
|
||||||
|
return $cmp if $cmp;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<isValidHostname>
|
||||||
|
|
||||||
|
This function returns true if it is passed a valid RFC 921 hostname,
|
||||||
|
false otherwise.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub isValidHostname
|
||||||
|
{
|
||||||
|
my $host_or_ip = shift;
|
||||||
|
|
||||||
|
unless (isValidIP($host_or_ip))
|
||||||
|
{
|
||||||
|
# It's not an IP address. Does it look like a hostname?
|
||||||
|
# FIXME: We could do a DNS lookup to be sure.
|
||||||
|
# (See RFC 921, "Domain Name System Implementation Schedule,"
|
||||||
|
# FIXME: Put this in a library.
|
||||||
|
unless ($host_or_ip =~ m{
|
||||||
|
# Must begin with an alphabetical character...
|
||||||
|
^[a-z]
|
||||||
|
# optionally followed by zero or more alphabetic characters,
|
||||||
|
# hyphens, periods and numbers...
|
||||||
|
[-a-z.0-9]*
|
||||||
|
(
|
||||||
|
# followed by one period...
|
||||||
|
\.
|
||||||
|
# and a repeat of the first pattern
|
||||||
|
[a-z]
|
||||||
|
[-a-z.0-9]*
|
||||||
|
)+
|
||||||
|
# which we can repeat one or more times, to the end of the
|
||||||
|
# string.
|
||||||
|
$
|
||||||
|
# Case insensitive.
|
||||||
|
}ix)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
=item I<isValidEmail>
|
||||||
|
|
||||||
|
This validation function validates an email address, using the
|
||||||
|
Mail::RFC822::Address module. Additionally, by default, it permits a local
|
||||||
|
address instead of a fully-qualified remote address, even checking the
|
||||||
|
existence of said user in the accounts db.
|
||||||
|
|
||||||
|
If you don't wish to permit local addresses, pass the permitlocal option as
|
||||||
|
false.
|
||||||
|
|
||||||
|
ie. esmith::util::isValidEmail($address, { permitlocal => 0 })
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub isValidEmail
|
||||||
|
{
|
||||||
|
my $address = shift;
|
||||||
|
my $hashref = shift || {};
|
||||||
|
my %defaults = ( permitlocal => 1 );
|
||||||
|
my %options = (%defaults, %$hashref);
|
||||||
|
|
||||||
|
if (Mail::RFC822::Address::valid($address))
|
||||||
|
{
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
# Permit a local address.
|
||||||
|
if ($address =~ /^[a-zA-Z][a-zA-Z0-9\._\-]*$/)
|
||||||
|
{
|
||||||
|
# Exception for 'admin' user. FIXME - I'd rather not hardcode this,
|
||||||
|
# but we can't permit email to all system users.
|
||||||
|
return TRUE if $address eq 'admin';
|
||||||
|
# Make sure the user exists.
|
||||||
|
my $accountsdb = esmith::AccountsDB->open_ro;
|
||||||
|
my $user = $accountsdb->get($address) || '';
|
||||||
|
unless (($user) && ($user->prop('type') eq 'user'))
|
||||||
|
{
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
Mitel Networks Corp.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
215
root/usr/share/perl5/vendor_perl/esmith/util/system.pm
Normal file
215
root/usr/share/perl5/vendor_perl/esmith/util/system.pm
Normal file
@ -0,0 +1,215 @@
|
|||||||
|
#----------------------------------------------------------------------
|
||||||
|
# Copyright 1999-2003 Mitel Networks Corporation
|
||||||
|
# This program is free software; you can redistribute it and/or
|
||||||
|
# modify it under the same terms as Perl itself.
|
||||||
|
#----------------------------------------------------------------------
|
||||||
|
|
||||||
|
package esmith::util::system;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
require Exporter;
|
||||||
|
our @ISA = qw(Exporter);
|
||||||
|
|
||||||
|
our @EXPORT_OK = qw(killall rsync rsync_ssh);
|
||||||
|
|
||||||
|
=for testing
|
||||||
|
use_ok('esmith::util::system', 'killall', 'rsync', 'rsync_ssh');
|
||||||
|
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
esmith::util::system - replacements/wrappers for system() commands
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use esmith::util::system qw(killall rsync rsync_ssh);
|
||||||
|
|
||||||
|
killall($signal, @commands);
|
||||||
|
rsync($src, $dest, @options);
|
||||||
|
rsync_ssh($src, $dest, $user, $ident, \@rsync_opts, \@ssh_opts);
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This is for common functions that would normally require a system(...)
|
||||||
|
command. Makes things easier to document, test and upgrade.
|
||||||
|
|
||||||
|
If you find yourself writing a system() command, consider putting it
|
||||||
|
in here.
|
||||||
|
|
||||||
|
|
||||||
|
=head2 Functions
|
||||||
|
|
||||||
|
These may be pure Perl functions or they may well just be wrappers
|
||||||
|
around system() commands.
|
||||||
|
|
||||||
|
Each can be imported on request.
|
||||||
|
|
||||||
|
=over 4
|
||||||
|
|
||||||
|
=item B<killall>
|
||||||
|
|
||||||
|
my $killed_something = killall($signal, @commands);
|
||||||
|
|
||||||
|
Sends a $signal to all of the named @commands. $signal can be
|
||||||
|
specified by name or number (so 1 or HUP for example, names are
|
||||||
|
prefered).
|
||||||
|
|
||||||
|
Returns true if something was killed, false otherwise.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
open(SCRATCH, ">scratch.exe") || die $!;
|
||||||
|
|
||||||
|
# XXX Irritating perl bug ends POD processing if it sees /^#!.*perl/
|
||||||
|
print SCRATCH sprintf <<'ENDING', '/usr/bin/perl';
|
||||||
|
#!%s -w
|
||||||
|
|
||||||
|
sleep 99;
|
||||||
|
ENDING
|
||||||
|
|
||||||
|
close SCRATCH;
|
||||||
|
|
||||||
|
END { unlink 'scratch.exe', 'scratch.out' }
|
||||||
|
|
||||||
|
chmod 0755, 'scratch.exe' || die $!;
|
||||||
|
my $pid = open(SCRATCH, "./scratch.exe |");
|
||||||
|
ok( $pid, 'scratch program started ok' );
|
||||||
|
|
||||||
|
ok( killall('USR1', 'scratch.exe'), 'killall returned properly' );
|
||||||
|
close SCRATCH; # so scratch.exe responds to the signal and exits
|
||||||
|
|
||||||
|
is( kill(9, $pid), 0, 'killall worked' );
|
||||||
|
|
||||||
|
# I can't actually think of a way to explicitly check this but it
|
||||||
|
# will make noise if it doesn't work.
|
||||||
|
ok( !killall('USR1', 'I_dont_exist_nope'),
|
||||||
|
'returned properly for killing nothing' );
|
||||||
|
ok( 1, 'killall is quiet when nothing is killed' );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub killall {
|
||||||
|
my($signal, @commands) = @_;
|
||||||
|
|
||||||
|
warn "You don't need a - on the signal" if $signal =~ /^-/;
|
||||||
|
|
||||||
|
my $killed_something =
|
||||||
|
system('/usr/bin/killall', '-q', "-$signal", @commands);
|
||||||
|
|
||||||
|
return !$killed_something;
|
||||||
|
}
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
|
=item B<rsync>
|
||||||
|
|
||||||
|
my $success = rsync($source, $destination, @options);
|
||||||
|
|
||||||
|
rsyncs the $source file or directory to the $destination. Any
|
||||||
|
@options are direct options to the rsync command.
|
||||||
|
|
||||||
|
rsync will be run --quiet by default.
|
||||||
|
|
||||||
|
Returns true if the rsync succeeds, false otherwise.
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use File::Compare;
|
||||||
|
my $src = '10e-smith-lib/db.t';
|
||||||
|
my $dest = '10e-smith-lib/db.t.copy';
|
||||||
|
rsync($src, $dest);
|
||||||
|
END { unlink $dest }
|
||||||
|
|
||||||
|
ok( -e $dest );
|
||||||
|
ok( compare($src, $dest) == 0, 'basic rsync copy' );
|
||||||
|
|
||||||
|
open(DEST, ">$dest" ) || die $!;
|
||||||
|
print DEST "Fooble\n";
|
||||||
|
close DEST;
|
||||||
|
|
||||||
|
# rsync in update-only mode. $dest is newer than $src and shouldn't
|
||||||
|
# be updated.
|
||||||
|
rsync($src, $dest, qw(--update));
|
||||||
|
|
||||||
|
ok( compare($src, $dest) == 1, 'rsync, update only' );
|
||||||
|
|
||||||
|
open(DEST, "$dest") || die $!;
|
||||||
|
my $data = join '', <DEST>;
|
||||||
|
close DEST;
|
||||||
|
|
||||||
|
is( $data, "Fooble\n" );
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
our $RSYNC_CMD = '/usr/bin/rsync';
|
||||||
|
sub rsync {
|
||||||
|
my($src, $dest, @options) = @_;
|
||||||
|
|
||||||
|
push @options, '--quiet';
|
||||||
|
return !system($RSYNC_CMD, @options, $src, $dest);
|
||||||
|
}
|
||||||
|
|
||||||
|
=item B<rsync_ssh>
|
||||||
|
|
||||||
|
my $success = rsync_ssh($src, $dest, $user, $ident, \@rsync_opts,
|
||||||
|
\@ssh_opts);
|
||||||
|
|
||||||
|
Like rsync() except it uses ssh. A typical call might be:
|
||||||
|
|
||||||
|
rsync_ssh('some.host:', 'some/file', 'someuser', 'some/.ssh/identity',
|
||||||
|
[qw(--archive --delete)]);
|
||||||
|
|
||||||
|
=begin testing
|
||||||
|
|
||||||
|
use File::Compare;
|
||||||
|
my $src = '10e-smith-lib/db.t';
|
||||||
|
my $dest = '10e-smith-lib/db.t.copy';
|
||||||
|
|
||||||
|
unlink $dest;
|
||||||
|
END { unlink $dest }
|
||||||
|
|
||||||
|
no warnings 'once';
|
||||||
|
my @args;
|
||||||
|
my $real_rsync = \&esmith::util::system::rsync;
|
||||||
|
local *esmith::util::system::rsync = sub {
|
||||||
|
@args = @_;
|
||||||
|
|
||||||
|
pop @_;
|
||||||
|
$real_rsync->(@_);
|
||||||
|
};
|
||||||
|
|
||||||
|
rsync_ssh($src, $dest, 'someone', 'some/ident', [qw(--update)], [qw(-C)]);
|
||||||
|
ok( -e $dest );
|
||||||
|
ok( compare($src, $dest) == 0 );
|
||||||
|
|
||||||
|
is($args[0], $src );
|
||||||
|
is($args[1], $dest );
|
||||||
|
is($args[2], '--update' );
|
||||||
|
is($args[3], "-e $esmith::util::system::SSH_CMD -l someone -i some/ident -C");
|
||||||
|
|
||||||
|
=end testing
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
our $SSH_CMD = '/usr/bin/ssh';
|
||||||
|
sub rsync_ssh {
|
||||||
|
my($src, $dest, $user, $ident, $rsync_opts, $ssh_opts) = @_;
|
||||||
|
|
||||||
|
$ssh_opts ||= [];
|
||||||
|
my $ssh_opt = join ' ', ('-e', $SSH_CMD, '-l', $user, '-i', $ident,
|
||||||
|
@$ssh_opts);
|
||||||
|
|
||||||
|
return rsync($src, $dest, @$rsync_opts, $ssh_opt);
|
||||||
|
}
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
Mitel Networks Corporation
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
1;
|
Loading…
Reference in New Issue
Block a user