Current File : //usr/share/webmin/lpadmin/lpadmin-lib.pl |
# lpadmin-lib.pl
# Functions for configuring and adding printers
BEGIN { push(@INC, ".."); };
use WebminCore;
&init_config();
do "$config{'print_style'}-lib.pl";
if ($config{'driver_style'}) {
do "$config{'driver_style'}-driver.pl";
}
else {
do "webmin-driver.pl";
}
%access = &get_module_acl();
$drivers_directory = "$module_config_directory/drivers";
# dev_name(file)
sub dev_name
{
local($i);
for($i=0; $i<@device_files; $i++) {
if ($device_files[$i] eq $_[0]) { return $device_names[$i]; }
}
return $_[0];
}
# has_ghostscript()
# Does this system have ghostscript installed?
sub has_ghostscript
{
return &has_command($config{'gs_path'});
}
# has_smbclient()
# Does this system have smbclient installed?
sub has_smbclient
{
return &has_command($config{'smbclient_path'});
}
# has_hpnp()
# Does this system have hpnp installed?
sub has_hpnp
{
return &has_command($config{'hpnp_path'});
}
# create_webmin_driver(&printer, &driver)
# lpadmin drivers are files in /etc/webmin/lpadmin/drivers. Each is a
# dynamically generated shell script which calls GS
sub create_webmin_driver
{
# check for non-driver
if ($_[1]->{'mode'} == 0) {
return undef;
}
elsif ($_[1]->{'mode'} == 2) {
return $_[1]->{'program'};
}
local($drv, $d, $gsdrv, $res, $perl);
&lock_file($drivers_directory);
mkdir($drivers_directory, 0755);
&unlock_file($drivers_directory);
$drv = "$drivers_directory/$_[0]->{'name'}";
# Find ghostscript driver
if ($_[1]->{'mode'} == 3) {
$gsdrv = 'uniprint';
}
else {
foreach $d (&list_webmin_drivers()) {
if ($d->[1] eq $_[1]->{'type'}) {
$gsdrv = $d->[0];
}
}
}
# Create script to call GS
&open_lock_tempfile(DRV, ">$drv");
&print_tempfile(DRV, "#!/bin/sh\n");
&print_tempfile(DRV, "# Name: $_[0]->{'name'}\n");
&print_tempfile(DRV, "# Type: ",$_[1]->{'upp'} ? 'uniprint'
: $_[1]->{'type'},"\n");
&print_tempfile(DRV, "# DPI: ",$_[1]->{'upp'} ? $_[1]->{'upp'}
: $_[1]->{'dpi'},"\n");
if ($gconfig{'ld_env'}) {
&print_tempfile(DRV, "$gconfig{'ld_env'}=$gconfig{'ld_path'}\n");
}
&print_tempfile(DRV, "PATH=$gconfig{'path'}\n");
if ($config{'gs_fontpath'}) {
&print_tempfile(DRV, "GS_FONTPATH=$config{'gs_fontpath'}\n");
}
if ($config{'gs_lib'}) {
&print_tempfile(DRV, "GS_LIB=$config{'gs_lib'}\n");
}
&print_tempfile(DRV, "export $gconfig{'ld_env'} PATH GS_FONTPATH GS_LIB\n");
$res = $_[1]->{'upp'} ? "\@$_[1]->{'upp'}.upp" :
$_[1]->{'dpi'} ? "-r$_[1]->{'dpi'}" : "";
$perl = &get_perl_path();
if ($config{'iface_arg'}) {
for($i=0; $i<$config{'iface_arg'}-1; $i++) {
&print_tempfile(DRV, "shift\n");
}
&print_tempfile(DRV, "cat \$* | $perl -e 'while(<STDIN>) { print if (!/^\\s*#####/); }' >/tmp/\$\$.gsin\n");
}
else {
&print_tempfile(DRV, "$perl -e 'while(<STDIN>) { print if (!/^\\s*#####/); }' >/tmp/\$\$.gsin\n");
}
&print_tempfile(DRV, "$config{'gs_path'} -sOutputFile=/tmp/\$\$.gs -dSAFER -sDEVICE=$gsdrv $res -dNOPAUSE /tmp/\$\$.gsin </dev/null >/dev/null 2>&1\n");
&print_tempfile(DRV, "rm /tmp/\$\$.gsin\n");
&print_tempfile(DRV, "cat /tmp/\$\$.gs\n");
&print_tempfile(DRV, "rm /tmp/\$\$.gs\n");
&close_tempfile(DRV);
if ($config{'iface_owner'}) {
&system_logged("chown '$config{'iface_owner'}' '$drv' >/dev/null 2>&1");
}
&system_logged("chmod '$config{'iface_perms'}' '$drv' >/dev/null 2>&1");
return $drv;
}
# create_webmin_windows_driver(&printer, &driver)
# Create an interface program that can print to a remote windows printer
# using some printer driver
sub create_webmin_windows_driver
{
local($drv, $prog);
&lock_file($drivers_directory);
mkdir($drivers_directory, 0755);
&unlock_file($drivers_directory);
$drv = "$drivers_directory/$_[0]->{'name'}.smb";
# Create script to call smbclient
&open_lock_tempfile(DRV, ">$drv");
&print_tempfile(DRV, "#!/bin/sh\n");
&print_tempfile(DRV, "# Name: $_[0]->{'name'}\n");
&print_tempfile(DRV, "# Server: $_[1]->{'server'}\n");
&print_tempfile(DRV, "# Share: $_[1]->{'share'}\n");
&print_tempfile(DRV, "# User: $_[1]->{'user'}\n");
&print_tempfile(DRV, "# Password: $_[1]->{'pass'}\n");
&print_tempfile(DRV, "# Workgroup: $_[1]->{'workgroup'}\n");
&print_tempfile(DRV, "# Program: $_[1]->{'program'}\n");
if ($gconfig{'ld_env'}) {
&print_tempfile(DRV, "$gconfig{'ld_env'}=$gconfig{'ld_path'}\n");
}
&print_tempfile(DRV, "PATH=$gconfig{'path'}\n");
&print_tempfile(DRV, "export $gconfig{'ld_env'} PATH\n");
if (!$_[1]->{'program'}) {
if ($config{'iface_arg'}) {
for($i=0; $i<$config{'iface_arg'}-1; $i++) {
&print_tempfile(DRV, "shift\n");
}
&print_tempfile(DRV, "cat \$* >/tmp/\$\$.smb\n");
}
else { &print_tempfile(DRV, "cat >/tmp/\$\$.smb\n"); }
}
else {
&print_tempfile(DRV, "$_[1]->{'program'} \"\$1\" \"\$2\" \"\$3\" \"\$4\" ",
"\"\$5\" \"\$6\" \"\$7\" \"\$8\" \"\$9\" ",
"\"\$10\" \"\$11\" \"\$12\" \"\$13\" >/tmp/\$\$.smb\n");
&system_logged("chmod a+rx '$_[1]->{'program'}'");
}
&print_tempfile(DRV, "$config{'smbclient_path'} '//$_[1]->{'server'}/$_[1]->{'share'}' ",
$_[1]->{'pass'} ? $_[1]->{'pass'} : "-N",
$_[1]->{'user'} ? " -U $_[1]->{'user'}" : "",
$_[1]->{'workgroup'} ? " -W $_[1]->{'workgroup'}" : "",
" -c \"print /tmp/\$\$.smb\"\n");
&print_tempfile(DRV, "rm /tmp/\$\$.smb\n");
&close_tempfile(DRV);
if ($config{'iface_owner'}) {
&system_logged("chown '$config{'iface_owner'}' '$drv' >/dev/null 2>&1");
}
&system_logged("chmod '$config{'iface_perms'}' '$drv' >/dev/null 2>&1");
return $drv;
}
# create_hpnp_driver(&printer, &driver)
# Create an interface program that can print to a hpnp server using some
# interface program
sub create_hpnp_driver
{
local($drv, $prog);
&lock_file($drivers_directory);
mkdir($drivers_directory, 0755);
&unlock_file($drivers_directory);
$drv = "$drivers_directory/$_[0]->{'name'}.hpnp";
# Create script to call hpnp
&open_lock_tempfile(DRV, ">$drv");
&print_tempfile(DRV, "#!/bin/sh\n");
&print_tempfile(DRV, "# Name: $_[0]->{'name'}\n");
&print_tempfile(DRV, "# Server: $_[1]->{'server'}\n");
&print_tempfile(DRV, "# Port: $_[1]->{'port'}\n");
&print_tempfile(DRV, "# Program: $_[1]->{'program'}\n");
if ($gconfig{'ld_env'}) {
&print_tempfile(DRV, "$gconfig{'ld_env'}=$gconfig{'ld_path'}\n");
}
&print_tempfile(DRV, "PATH=$gconfig{'path'}\n");
&print_tempfile(DRV, "export $gconfig{'ld_env'} PATH\n");
if (!$_[1]->{'program'}) {
if ($config{'iface_arg'}) {
for($i=0; $i<$config{'iface_arg'}-1; $i++) {
&print_tempfile(DRV, "shift\n");
}
&print_tempfile(DRV, "cat \$* >/tmp/\$\$.hpnp\n");
}
else { &print_tempfile(DRV, "cat >/tmp/\$\$.hpnp\n"); }
}
else {
&print_tempfile(DRV, "$_[1]->{'program'} \"\$1\" \"\$2\" \"\$3\" \"\$4\" ",
"\"\$5\" \"\$6\" \"\$7\" \"\$8\" \"\$9\" ",
"\"\$10\" \"\$11\" \"\$12\" \"\$13\" >/tmp/\$\$.hpnp\n");
&system_logged("chmod a+rx '$_[1]->{'program'}'");
}
&print_tempfile(DRV, "$config{'hpnp_path'} -x $_[1]->{'server'}",
$_[1]->{'port'} ? " -p $_[1]->{'port'}" : "",
" /tmp/\$\$.hpnp\n");
&print_tempfile(DRV, "rm /tmp/\$\$.hpnp\n");
&close_tempfile(DRV);
if ($config{'iface_owner'}) {
&system_logged("chown $config{'iface_owner'} $drv >/dev/null 2>&1");
}
&system_logged("chmod '$config{'iface_perms'}' '$drv' >/dev/null 2>&1");
&unlock_file($drv);
return $drv;
}
# is_webmin_driver(path)
# Returns a structure of driver information
sub is_webmin_driver
{
local($l, $i, $u, $desc);
if (!$_[0]) {
return { 'mode' => 0,
'desc' => 'None' };
}
if (&has_ghostscript()) {
open(DRV, "<".$_[0]);
for($i=0; $i<4; $i++) { $l .= <DRV>; }
close(DRV);
if ($l =~ /# Name: (.*)\n# Type: (.*)\n# DPI: (.*)\n/) {
if ($2 eq 'uniprint') {
local $upp = $3;
foreach $u (&list_uniprint()) {
$desc = $u->[1] if ($u->[0] eq $upp);
}
$desc =~ s/,.*$//;
return { 'mode' => 3,
'upp' => $upp,
'desc' => $desc ? $desc : $upp };
}
else {
return { 'type' => $2,
'dpi' => $3,
'mode' => 1,
'desc' => $3 ? "$2 ($3 DPI)" : $2 };
}
}
}
return { 'desc' => $_[0],
'prog' => $_[0],
'mode' => 2 };
}
# is_webmin_windows_driver(path)
# Returns a structure containing information about a windows driver, or undef
# Returns the server, share, username, password, workgroup, program
# if path is a webmin windows driver
sub is_webmin_windows_driver
{
local($i, $l);
if (!&has_smbclient()) { return undef; }
open(DRV, "<".$_[0]);
for($i=0; $i<8; $i++) { $l .= <DRV>; }
close(DRV);
if ($l =~ /# Name: (.*)\n# Server: (.*)\n# Share: (.*)\n# User: (.*)\n# Password: (.*)\n# Workgroup: (.*)\n# Program: (.*)\n/) {
return { 'server' => $2,
'share' => $3,
'user' => $4,
'pass' => $5,
'workgroup' => $6,
'program' => $7 };
}
elsif ($l =~ /# Name: (.*)\n# Server: (.*)\n# Share: (.*)\n# User: (.*)\n# Password: (.*)\n# Program: (.*)\n/) {
return { 'server' => $2,
'share' => $3,
'user' => $4,
'pass' => $5,
'program' => $7 };
}
else { return undef; }
}
# delete_webmin_driver(name)
# Delete the drivers for some printer
sub delete_webmin_driver
{
&lock_file("$drivers_directory/$_[0]");
&lock_file("$drivers_directory/$_[0].smb");
&lock_file("$drivers_directory/$_[0].hpnp");
unlink("$drivers_directory/$_[0]");
unlink("$drivers_directory/$_[0].smb");
unlink("$drivers_directory/$_[0].hpnp");
&unlock_file("$drivers_directory/$_[0]");
&unlock_file("$drivers_directory/$_[0].smb");
&unlock_file("$drivers_directory/$_[0].hpnp");
}
# is_hpnp_driver(path, &printer)
# Returns a structure of hpnp details if path is a webmin hpnp driver
sub is_hpnp_driver
{
local($i, $l);
if (!&has_hpnp()) { return undef; }
open(DRV, "<".$_[0]);
for($i=0; $i<5; $i++) { $l .= <DRV>; }
close(DRV);
if ($l =~ /# Name: (.*)\n# Server: (.*)\n# Port: (.*)\n# Program: (.*)\n/) {
return { 'server' => $2,
'port' => $3,
'program' => $4 };
}
else { return undef; }
}
# webmin_driver_input(&printer, &driver)
sub webmin_driver_input
{
local ($prn, $drv) = @_;
printf "<tr> <td><input type=radio name=drv value=0 %s> %s</td>\n",
$drv->{'mode'} == 0 ? "checked" : "", $text{'webmin_none'};
print "<td>($text{'webmin_nonemsg'})</td> </tr>\n";
printf "<tr> <td><input type=radio name=drv value=2 %s> %s</td>\n",
$drv->{'mode'} == 2 ? "checked" : "", $text{'webmin_prog'};
printf "<td><input name=iface value=\"%s\" size=35></td> </tr>\n",
$drv->{'mode'} == 2 ? $drv->{'prog'} : "";
if (&has_ghostscript()) {
local $out = &backquote_command("$config{'gs_path'} -help 2>&1", 1);
if ($out =~ /Available devices:\n((\s+.*\n)+)/i) {
print "<tr> <td valign=top>\n";
printf "<input type=radio name=drv value=1 %s>\n",
$drv->{'mode'} == 1 ? "checked" : "";
print "$text{'webmin_driver'}</td> <td valign=top>";
foreach $d (split(/\s+/, $1)) { $drvsupp{$d}++; }
print "<select name=driver size=7>\n";
foreach $d (&list_webmin_drivers()) {
if ($drvsupp{$d->[0]}) {
printf "<option value='%s' %s>%s (%s)</option>\n",
$d->[1],
$d->[1] eq $drv->{'type'} ? "selected" : "",
$d->[1], $d->[0];
}
}
print "</select> ";
print "<select name=dpi size=7>\n";
printf "<option value=\"\" %s>Default</option>\n",
$drv->{'dpi'} ? "" : "selected";
foreach $d (75, 100, 150, 200, 300, 600, 720, 1440) {
printf "<option value=\"$d\" %s>$d DPI</option>\n",
$drv->{'dpi'} == $d ? "selected" : "";
}
print "</select></td> </tr>\n";
if ($drvsupp{'uniprint'}) {
print "<tr> <td valign=top>\n";
printf "<input type=radio name=drv value=3 %s>\n",
$drv->{'mode'} == 3 ? "checked" : "";
print "$text{'webmin_uniprint'}</td> <td valign=top>";
print "<select name=uniprint size=5>\n";
foreach $u (&list_uniprint()) {
printf "<option value=%s %s>%s</option>\n",
$u->[0],
$u->[0] eq $drv->{'upp'} ? 'selected' : '',
$u->[1];
}
print "</select></td> </tr>\n";
}
}
else {
print "<tr> <td colspan=2>",
&text('webmin_edrivers', "<tt>$config{'gs_path'}</tt>"),
"</td> </tr>\n";
}
}
elsif ($config{'gs_path'}) {
print "<tr> <td colspan=2>",
&text('webmin_egs', "<tt>$config{'gs_path'}</tt>"),
"</td> </tr>\n";
}
return undef;
}
# parse_webmin_driver()
# Parse driver selection from %in and return a driver structure
sub parse_webmin_driver
{
if ($in{'drv'} == 0) {
return { 'mode' => 0 };
}
elsif ($in{'drv'} == 2) {
my @iface = split(/\s+/, $in{'iface'});
-x $iface[0] || &error(&text('webmin_edriver', $iface[0]));
return { 'mode' => 2,
'program' => $in{'iface'} };
}
elsif ($in{'drv'} == 1) {
return { 'mode' => 1,
'type' => $in{'driver'},
'dpi' => $in{'dpi'} };
}
elsif ($in{'drv'} == 3) {
return { 'mode' => 3,
'upp' => $in{'uniprint'} };
}
}
# list_webmin_drivers()
sub list_webmin_drivers
{
local(@rv, $_);
open(DRIVERS, "<$module_root_directory/drivers");
while(<DRIVERS>) {
/^(\S+)\s+(.*)/;
push(@rv, [ $1, $2 ]);
}
close(DRIVERS);
return @rv;
}
# can_edit_printer(printer)
sub can_edit_printer
{
foreach $p (split(/\s+/, $access{'printers'})) {
return 1 if ($p eq '*' || $p eq $_[0]);
}
return 0;
}
# can_edit_jobs(printer, user)
sub can_edit_jobs
{
local $rv = 0;
if ($access{'cancel'} == 1) {
$rv = 1;
}
elsif ($access{'cancel'} == 0) {
$rv = 0;
}
else {
foreach $p (split(/\s+/, $access{'jobs'})) {
$rv = 1 if ($p eq $_[0]);
}
}
if ($rv) {
if ($access{'user'} eq '*') {
return 1;
}
elsif ($access{'user'} eq $_[1]) {
return 1;
}
elsif (!$access{'user'} && $remote_user eq $_[1]) {
return 1;
}
}
return 0;
}
# list_uniprint()
# Returns a list of uniprint drivers support by the installed ghostscript
sub list_uniprint
{
local (@rv, $f, $d);
local $out = &backquote_command("$config{'gs_path'} -help 2>&1", 1);
if ($out =~ /Search path:\n((\s+.*\n)+)/i) {
foreach $d (split(/\s+/, $1)) {
next if ($d !~ /^\//);
opendir(DIR, $d);
while($f = readdir(DIR)) {
next if ($f !~ /^(.*)\.upp$/);
local $upp = $1;
open(UPP, "<$d/$f");
local $line = <UPP>;
close(UPP);
next if ($line !~ /upModel="(.*)"/i);
push(@rv, [ $upp, $1 ]);
}
closedir(DIR);
}
}
return sort { $a->[0] cmp $b->[0] } @rv;
}
sub log_info
{
local ($drv, $wdrv, $hdrv);
if (!$webmin_windows_driver) {
$wdrv = &is_webmin_windows_driver($_[0]->{'iface'}, $_[0]);
}
$wdrv = &is_windows_driver($_[0]->{'iface'}, $_[0]) if (!$wdrv);
$hdrv = &is_hpnp_driver($_[0]->{'iface'}, $_[0]);
local $iface = $wdrv ? $wdrv->{'program'} :
$hdrv ? $hdrv->{'program'} : $_[0]->{'iface'};
if (!$webmin_print_driver) {
$drv = &is_webmin_driver($iface, $_[0]);
}
$drv = &is_driver($iface, $_[0])
if ($drv->{'mode'} == 0 || $drv->{'mode'} == 2);
$drv->{'desc'} =~ s/\([^\)]+\)$//;
return { 'driver' => $drv->{'desc'},
'mode' => $drv->{'mode'},
'dest' => $wdrv ? "\\\\$wdrv->{'server'}\\$wdrv->{'share'}" :
$hdrv ? "HPNP $hdrv->{'server'}:$hdrv->{'port'}" :
$_[0]->{'rhost'} ? "$_[0]->{'rhost'}:$_[0]->{'rqueue'}" :
$_[0]->{'dhost'} ? "$_[0]->{'dhost'}:$_[0]->{'dport'}" :
&dev_name($_[0]->{'dev'}) };
}
# parse_cups_ppd(file)
# Converts a CUPS-style .ppd file into a hash of names and values
sub parse_cups_ppd
{
local ($file) = @_;
local %ppd;
if ($file =~ /\.gz$/) {
open(PPD, "gunzip -c ".quotemeta($file)." |");
}
else {
open(PPD, "<".$file);
}
while(<PPD>) {
if (/^\s*\*(\S+):\s*"(.*)"/ || /^\s*\*(\S+):\s*(\S+)/) {
$ppd{$1} = $2;
}
elsif (/^\s*\*(\S+)\s+(\S+)\/([^:]+):/) {
$ppd{$1}->{$2} = $3 if (!defined($ppd{$1}->{$2}));
}
}
close(PPD);
return \%ppd;
}
# list_cluster_servers()
# Returns a list of servers on which printers are managed
sub list_cluster_servers
{
&foreign_require("servers", "servers-lib.pl");
local %ids = map { $_, 1 } split(/\s+/, $config{'servers'});
return grep { $ids{$_->{'id'}} } &servers::list_servers();
}
# add_cluster_server(&server)
sub add_cluster_server
{
local @sids = split(/\s+/, $config{'servers'});
$config{'servers'} = join(" ", @sids, $_[0]->{'id'});
&save_module_config();
}
# delete_cluster_server(&server)
sub delete_cluster_server
{
local @sids = split(/\s+/, $config{'servers'});
$config{'servers'} = join(" ", grep { $_ != $_[0]->{'id'} } @sids);
&save_module_config();
}
# server_name(&server)
sub server_name
{
return $_[0]->{'desc'} ? $_[0]->{'desc'} : $_[0]->{'host'};
}
# save_printer_cluster(new, &printer, &driver, &connection, webmin-driver, mode)
# Creates or updates the specified printer on all cluster hosts, and returns a
# list of error messages
sub save_on_cluster
{
return ( ) if (!$config{'servers'});
return ( ) if (&is_readonly_mode());
local ($new, $prn, $drv, $conn, $webmin, $mode) = @_;
&remote_error_setup(\&slave_error_handler);
local $slave;
local @slaveerrs;
foreach $slave (&list_cluster_servers()) {
# Connect to server
$slave_error = undef;
&remote_foreign_require($slave, "lpadmin", "lpadmin-lib.pl");
if ($slave_error) {
push(@slaveerrs, [ $slave, $slave_error ]);
next;
}
# Create the driver and the printer
local $err = &remote_foreign_call($slave,
"lpadmin", "save_printer_and_driver",
$new, $prn, $drv, $conn, $webmin, $mode);
if ($slave_error) {
push(@slaveerrs, [ $slave, $slave_error ]);
}
elsif ($err == 1) {
push(@slaveerrs, [ $slave, &text('save_edup', $prn->{'name'}) ]);
}
elsif ($err == 2) {
push(@slaveerrs, [ $slave, $text{'save_evalid'} ]);
}
elsif ($err == 3) {
push(@slaveerrs, [ $slave, &text('save_egone', $prn->{'name'}) ]);
}
}
return @slaveerrs;
}
# delete_on_cluster(&printer)
# Deletes the specified printer on all cluster hosts, and returns a list of
# error messages.
sub delete_on_cluster
{
return ( ) if (!$config{'servers'});
return ( ) if (&is_readonly_mode());
local ($prn) = @_;
&remote_error_setup(\&slave_error_handler);
local $slave;
local @slaveerrs;
foreach $slave (&list_cluster_servers()) {
# Connect to server
$slave_error = undef;
&remote_foreign_require($slave, "lpadmin", "lpadmin-lib.pl");
if ($slave_error) {
push(@slaveerrs, [ $slave, $slave_error ]);
next;
}
# Call the delete function
local $err = &remote_foreign_call($slave,
"lpadmin", "delete_printer_and_driver", $prn);
if ($slave_error) {
push(@slaveerrs, [ $slave, $slave_error ]);
}
elsif ($err == 3) {
push(@slaveerrs, [ $slave, &text('save_egone', $prn->{'name'}) ]);
}
}
return @slaveerrs;
}
# save_printer_and_driver(new, &printer, &driver, &connection, webmin-driver, mode)
# Attempts to setup or modify a printer and driver. Returns 0 if OK, 1 if the
# printer already exists, 2 if some print system error occurred, or 3 if it
# doesn't exist but should.
sub save_printer_and_driver
{
local ($new, $prn, $drv, $conn, $webmin, $mode) = @_;
if ($new && &get_printer($prn->{'name'})) {
return 1;
}
elsif (!$new && !&get_printer($prn->{'name'})) {
return 2;
}
local $dfunc = $webmin ? \&create_webmin_driver : \&create_driver;
if ($mode <= 2 || $mode == 5) {
# Device, file or LPR host
$prn->{'iface'} = &$dfunc($prn, $drv);
}
elsif ($mode == 3) {
# Windows server
$prn->{'dev'} = "/dev/null";
$prn->{'iface'} = $webmin ? &create_webmin_windows_driver($prn, $conn)
: &create_windows_driver($prn, $conn);
}
elsif ($mode == 4) {
# HPNP server
$prn->{'dev'} = "/dev/null";
$prn->{'iface'} = &create_hpnp_driver($prn, $conn);
}
# Call os-specific validation function
if (defined(&validate_printer)) {
local $err = &validate_printer($prn);
return 2 if ($err);
}
# Actually create or update it
if ($new) {
&create_printer($prn);
}
else {
&modify_printer($prn);
}
&system_logged("$config{'apply_cmd'} >/dev/null 2>&1 </dev/null")
if ($config{'apply_cmd'});
return 0;
}
# delete_printer_and_driver(&printer)
# Deletes a printer, returning 1 if it could not be found, or 0 if everything
# went OK.
sub delete_printer_and_driver
{
local ($prn) = @_;
&delete_printer($prn->{'name'});
&delete_driver($prn->{'name'});
&delete_webmin_driver($prn->{'name'});
}
sub slave_error_handler
{
$slave_error = $_[0];
}
# delete_from_acls(name)
# Remove some named printer from all ACLs
sub delete_from_acls
{
local ($name) = @_;
local $wusers;
&read_acl(undef, \%wusers);
foreach my $u (keys %wusers) {
my %uaccess = &get_module_acl($u);
if ($uaccess{'printers'} ne '*') {
$uaccess{'printers'} =
join(' ', grep { $_ ne $name }
split(/\s+/, $uaccess{'printers'}));
&save_module_acl(\%uaccess, $u);
}
}
}
1;