xref: /illumos-gate/usr/src/cmd/print/scripts/lpadmin (revision 0a44ef6d9afbfe052a7e975f55ea0d2954b62a82)
1b51e021dSjacobs#!/bin/ksh
27c478bd9Sstevel@tonic-gate#
37c478bd9Sstevel@tonic-gate# CDDL HEADER START
47c478bd9Sstevel@tonic-gate#
57c478bd9Sstevel@tonic-gate# The contents of this file are subject to the terms of the
6355b4669Sjacobs# Common Development and Distribution License (the "License").
7355b4669Sjacobs# You may not use this file except in compliance with the License.
87c478bd9Sstevel@tonic-gate#
97c478bd9Sstevel@tonic-gate# You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
107c478bd9Sstevel@tonic-gate# or http://www.opensolaris.org/os/licensing.
117c478bd9Sstevel@tonic-gate# See the License for the specific language governing permissions
127c478bd9Sstevel@tonic-gate# and limitations under the License.
137c478bd9Sstevel@tonic-gate#
147c478bd9Sstevel@tonic-gate# When distributing Covered Code, include this CDDL HEADER in each
157c478bd9Sstevel@tonic-gate# file and include the License file at usr/src/OPENSOLARIS.LICENSE.
167c478bd9Sstevel@tonic-gate# If applicable, add the following below this CDDL HEADER, with the
177c478bd9Sstevel@tonic-gate# fields enclosed by brackets "[]" replaced with your own identifying
187c478bd9Sstevel@tonic-gate# information: Portions Copyright [yyyy] [name of copyright owner]
197c478bd9Sstevel@tonic-gate#
207c478bd9Sstevel@tonic-gate# CDDL HEADER END
217c478bd9Sstevel@tonic-gate#
227c478bd9Sstevel@tonic-gate#
23355b4669Sjacobs# Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
247c478bd9Sstevel@tonic-gate# Use is subject to license terms.
257c478bd9Sstevel@tonic-gate#
267c478bd9Sstevel@tonic-gate# ident	"%Z%%M%	%I%	%E% SMI"
277c478bd9Sstevel@tonic-gate#
287c478bd9Sstevel@tonic-gatePATH=/bin:/usr/bin:/usr/sbin export PATH
297c478bd9Sstevel@tonic-gate
307c478bd9Sstevel@tonic-gateTEXTDOMAIN="SUNW_OST_OSCMD"
317c478bd9Sstevel@tonic-gateexport TEXTDOMAIN
327c478bd9Sstevel@tonic-gate
337c478bd9Sstevel@tonic-gateLPSET=/usr/bin/lpset
347c478bd9Sstevel@tonic-gateLPGET=/usr/bin/lpget
35*0a44ef6dSjacobsLPSTAT=/usr/bin/lpstat
36b51e021dSjacobsLPADMIN=/usr/lib/lp/local/lpadmin
377c478bd9Sstevel@tonic-gate
38b51e021dSjacobsHOST=$(/bin/uname -n)
397c478bd9Sstevel@tonic-gateexit_code=0
407c478bd9Sstevel@tonic-gate
417c478bd9Sstevel@tonic-gateusage() {
427c478bd9Sstevel@tonic-gate	gettext "Usage:\n" 1>&2
437c478bd9Sstevel@tonic-gate	gettext "	lpadmin -p (printer) (options)\n" 1>&2
447c478bd9Sstevel@tonic-gate	gettext "	lpadmin -x (dest)\n" 1>&2
457c478bd9Sstevel@tonic-gate	gettext "	lpadmin -d (dest)\n" 1>&2
467c478bd9Sstevel@tonic-gate	gettext "	lpadmin -S print-wheel -A alert-type [ -W minutes ]\n" 1>&2
477c478bd9Sstevel@tonic-gate	gettext "		[ -Q requests ]\n" 1>&2
487c478bd9Sstevel@tonic-gate	gettext "	lpadmin -M -f form-name [ -a [ -o filebreak ]\n" 1>&2
497c478bd9Sstevel@tonic-gate	gettext "		[ -t tray-number ]]\n" 1>&2
507c478bd9Sstevel@tonic-gate	exit 1
517c478bd9Sstevel@tonic-gate}
527c478bd9Sstevel@tonic-gate
53b51e021dSjacobs# echo names in ${1} that are not in ${2}
54b51e021dSjacobsmissing() {
55b51e021dSjacobs	for i in ${1} ; do
56b51e021dSjacobs		MATCHED=0
57b51e021dSjacobs		for j in ${2} ; do
58b51e021dSjacobs			if [[ $i = $j ]] ; then
59b51e021dSjacobs				MATCHED=1
607c478bd9Sstevel@tonic-gate			fi
617c478bd9Sstevel@tonic-gate		done
62b51e021dSjacobs		if [[ $MATCHED == 0 ]] ; then
63b51e021dSjacobs			echo $i
647c478bd9Sstevel@tonic-gate		fi
657c478bd9Sstevel@tonic-gate	done
667c478bd9Sstevel@tonic-gate}
677c478bd9Sstevel@tonic-gate
68b51e021dSjacobs# create a filter table for LP service
69b51e021dSjacobslp_config_filters() {
70b51e021dSjacobs	if [[ ! -f /etc/lp/filter.table ]] ; then
71b51e021dSjacobs		cd /etc/lp/fd ; for filter in *.fd ; do
72b51e021dSjacobs			/usr/sbin/lpfilter \
73b51e021dSjacobs				-f $(/usr/bin/basename $filter .fd) \
74b51e021dSjacobs				-F $filter
75b51e021dSjacobs		done
76b51e021dSjacobs	fi
77b51e021dSjacobs}
78b51e021dSjacobs
79b51e021dSjacobs# enable/disable LP related service(s)
80b51e021dSjacobslp_config_service() {	# (enable | disable)
81b51e021dSjacobs	svcadm ${1} -s svc:/application/print/server:default
82b51e021dSjacobs	# svcadm ${1} -s svc:/application/print/rfc1179:default
83b51e021dSjacobs	# svcadm ${1} -s svc:/application/print/ipp-listener:default
84b51e021dSjacobs}
85b51e021dSjacobs
86b51e021dSjacobs# synchronize printers.conf with LP configuration changes
87b51e021dSjacobslp_config_sync_pconf() {	# (pre) (post)
88b51e021dSjacobs	if [[ "${1}" != "${2}" ]] ; then
89b51e021dSjacobs		ADDED=$(missing "${2}" "${1}")
90b51e021dSjacobs		REMOVED=$(missing "${1}" "${2}")
91b51e021dSjacobs
92b51e021dSjacobs		lp_server=${server:-${HOST}}
93b51e021dSjacobs		for DEST in ${ADDED} ; do
94b51e021dSjacobs			lp_uri="ipp://${lp_server}/printers/${DEST}"
95b51e021dSjacobs			lp_bsdaddr="${lp_server},${DEST},Solaris"
96b51e021dSjacobs			${LPSET} -n system \
97b51e021dSjacobs				-a "printer-uri-supported=${lp_uri}" \
98b51e021dSjacobs				-a "bsdaddr=${lp_bsdaddr}" \
99b51e021dSjacobs			 	${DEST} 2>/dev/null
100b51e021dSjacobs		done
101b51e021dSjacobs
102b51e021dSjacobs		for DEST in ${REMOVED} ; do
103b51e021dSjacobs			${LPSET} -n system -x ${DEST} 2>/dev/null
104b51e021dSjacobs		done
105b51e021dSjacobs	fi
106b51e021dSjacobs}
107b51e021dSjacobs
108b51e021dSjacobs# Delete all destinations in printers.conf
109b51e021dSjacobsdelete_all() {
110b51e021dSjacobs	for DEST in $(lpget -n system list | egrep -e '.+:$' | sed -e 's/://')
111b51e021dSjacobs	do
112b51e021dSjacobs		${LPSET} -n system -x ${DEST}
113b51e021dSjacobs		status=$?
114b51e021dSjacobs	done
115b51e021dSjacobs}
116b51e021dSjacobs
117b51e021dSjacobs#
118b51e021dSjacobs# Execution begins here
119b51e021dSjacobs#
120b51e021dSjacobs
121b51e021dSjacobs# be sure that we can run lpset and lpget
122b51e021dSjacobsif [[ ! -x ${LPSET} || ! -x ${LPGET} ]] ; then
123b51e021dSjacobs	gettext "lpadmin: System error; cannot set default printer\n" 1>&2
124b51e021dSjacobs	exit 2
125b51e021dSjacobsfi
126b51e021dSjacobs
127b51e021dSjacobsif [[ $# -lt 1 ]] ; then
1287c478bd9Sstevel@tonic-gate	usage
1297c478bd9Sstevel@tonic-gate	exit 1
1307c478bd9Sstevel@tonic-gatefi
1317c478bd9Sstevel@tonic-gate
1327c478bd9Sstevel@tonic-gate# Deal with the -d option independently since getopts does not handle
1337c478bd9Sstevel@tonic-gate# options that may or may not have arguments
1347c478bd9Sstevel@tonic-gate#
135b51e021dSjacobsif [[ ${1} = "-d" ]] ; then
136b51e021dSjacobs	if [[ $# -eq 1 ]] ; then	# remove the "default"
137b51e021dSjacobs		${LPGET} -n system _default >/dev/null 2>&1
138b51e021dSjacobs		exit_code=$?
1397c478bd9Sstevel@tonic-gate
140b51e021dSjacobs		if [[ ${exit_code} -eq 0 ]] ; then
141b51e021dSjacobs			${LPSET} -n system -x _default
142b51e021dSjacobs			exit_code=$?
143b51e021dSjacobs		else	# no default, nothing to do
144b51e021dSjacobs			exit_code=0
145b51e021dSjacobs		fi
146b51e021dSjacobs	elif [[ $# -eq 2 ]] ; then	# add/change the "default"
147b51e021dSjacobs		${LPGET} -k bsdaddr ${2} >/dev/null 2>&1
148b51e021dSjacobs		exit_code=$?
149b51e021dSjacobs
150b51e021dSjacobs		if [[ $exit_code -eq 0 ]] ; then
151b51e021dSjacobs			${LPSET} -n system -a "use=${2}" _default
152b51e021dSjacobs			exit_code=$?
153b51e021dSjacobs		else	# can't set default to an unconfigured printer
154b51e021dSjacobs			gettext "${2}: undefined printer\n" 1>&1
155b51e021dSjacobs		fi
156b51e021dSjacobs	else				# invalid usage
1577c478bd9Sstevel@tonic-gate		usage
1587c478bd9Sstevel@tonic-gate		exit 1
1597c478bd9Sstevel@tonic-gate	fi
1607c478bd9Sstevel@tonic-gate
1617c478bd9Sstevel@tonic-gate	exit ${exit_code}
1627c478bd9Sstevel@tonic-gatefi
1637c478bd9Sstevel@tonic-gate
1647c478bd9Sstevel@tonic-gate#		Strip off legal options
1657c478bd9Sstevel@tonic-gatewhile getopts "A:ac:D:e:f:F:H:hi:I:lm:Mn:o:p:Q:r:S:s:T:u:U:v:W:x:t:P:" arg
1667c478bd9Sstevel@tonic-gatedo
1677c478bd9Sstevel@tonic-gate	case $arg in
1687c478bd9Sstevel@tonic-gate	D)
1697c478bd9Sstevel@tonic-gate		description="${OPTARG}"
1707c478bd9Sstevel@tonic-gate	;;
1717c478bd9Sstevel@tonic-gate	p)
172b51e021dSjacobs		if [[ -n "${delete}" ]] ; then
1737c478bd9Sstevel@tonic-gate			usage
1747c478bd9Sstevel@tonic-gate		fi
1757c478bd9Sstevel@tonic-gate		printer=${OPTARG}
1767c478bd9Sstevel@tonic-gate	;;
1777c478bd9Sstevel@tonic-gate	s)
1787c478bd9Sstevel@tonic-gate		server=${OPTARG}
1797c478bd9Sstevel@tonic-gate	;;
1807c478bd9Sstevel@tonic-gate	v|U)
1817c478bd9Sstevel@tonic-gate		device=${OPTARG}
182b51e021dSjacobs		if [[ ! -n "${server}" ]] ; then
1835c88ba20Swendyp			server=${HOST}
1845c88ba20Swendyp		fi
185*0a44ef6dSjacobs		local="true"
1867c478bd9Sstevel@tonic-gate	;;
1877c478bd9Sstevel@tonic-gate	x)
188b51e021dSjacobs		if [[ -n "${printer}" || -n "${server}" || \
189b51e021dSjacobs		     -n "${device}" || -n "${description}" ]] ; then
1907c478bd9Sstevel@tonic-gate			usage
1917c478bd9Sstevel@tonic-gate		fi
1927c478bd9Sstevel@tonic-gate		delete=${OPTARG}
1937c478bd9Sstevel@tonic-gate		printer=${OPTARG}
194b51e021dSjacobs		if [[ ${printer} = "all" ]] ; then
1957c478bd9Sstevel@tonic-gate			local="true"
1967c478bd9Sstevel@tonic-gate		fi
1977c478bd9Sstevel@tonic-gate	;;
1987c478bd9Sstevel@tonic-gate	S|M|A)
1997c478bd9Sstevel@tonic-gate		local="true"
2007c478bd9Sstevel@tonic-gate	;;
2017c478bd9Sstevel@tonic-gate	c)
2027c478bd9Sstevel@tonic-gate		class=${OPTARG}
2037c478bd9Sstevel@tonic-gate		local="true"
204b51e021dSjacobs		if [[ ! -f ${LPGET} ]] ; then
2057c478bd9Sstevel@tonic-gate			gettext "lpadmin: System error; cannot set class\n " 1>&2
2067c478bd9Sstevel@tonic-gate			exit 2
2077c478bd9Sstevel@tonic-gate		fi
2087c478bd9Sstevel@tonic-gate
2097c478bd9Sstevel@tonic-gate		${LPGET} "${class}" > /dev/null 2>&1
2107c478bd9Sstevel@tonic-gate		lpget_class=$?
211b51e021dSjacobs		if [[ ${lpget_class} -eq 0 && ! -r /etc/lp/classes/"${class}" ]] ; then
2127c478bd9Sstevel@tonic-gate			gettext "lpadmin: ERROR: Can't create class ${class}.\n" 1>&2
2137c478bd9Sstevel@tonic-gate			gettext "           TO FIX: This is an existing printer name;\n" 1>&2
2147c478bd9Sstevel@tonic-gate			gettext "                   choose another name.\n" 1>&2
2157c478bd9Sstevel@tonic-gate			exit 1
2167c478bd9Sstevel@tonic-gate		fi
2177c478bd9Sstevel@tonic-gate	;;
2187c478bd9Sstevel@tonic-gate	r)
2197c478bd9Sstevel@tonic-gate		local="true"
2207c478bd9Sstevel@tonic-gate	;;
2217c478bd9Sstevel@tonic-gate	esac
2227c478bd9Sstevel@tonic-gatedone
2237c478bd9Sstevel@tonic-gate
2247c478bd9Sstevel@tonic-gate#
2257c478bd9Sstevel@tonic-gate# We don't have anything to do; let user know and bail
2267c478bd9Sstevel@tonic-gate#
227b51e021dSjacobsif [[ ! -n "${printer}" && ! -n "${delete}" && ! -n "${local}" ]] ; then
2287c478bd9Sstevel@tonic-gate	gettext "lpadmin: ERROR: Nothing to do.\n" 1>&2
2297c478bd9Sstevel@tonic-gate	gettext "        TO FIX: You must give one of these options:\n" 1>&2
2307c478bd9Sstevel@tonic-gate	gettext "		      -p, -d, -x -S\n" 1>&2
2317c478bd9Sstevel@tonic-gate	exit 1
2327c478bd9Sstevel@tonic-gatefi
2337c478bd9Sstevel@tonic-gate
2347c478bd9Sstevel@tonic-gate#
2357c478bd9Sstevel@tonic-gate#       Printer does not exist
2367c478bd9Sstevel@tonic-gate#       To be consistent with 2.5, assume adding local printer
2377c478bd9Sstevel@tonic-gate#
238b51e021dSjacobsif [[ ! -n "${device}" && ! -n "${server}" && ! -n "${delete}" && \
239b51e021dSjacobs	  ! -n "${local}" ]] ; then
2407c478bd9Sstevel@tonic-gate	${LPGET} "${printer}" > /dev/null 2>&1
2417c478bd9Sstevel@tonic-gate	lpget_stat=$?
242b51e021dSjacobs	if [[ ${lpget_stat} -ne 0 ]] ; then
2437c478bd9Sstevel@tonic-gate		gettext "lpadmin: ERROR: Missing -U or -v option.\n" 1>&2
2447c478bd9Sstevel@tonic-gate		gettext "           TO FIX: Local printers must have\n" 1>&2
2457c478bd9Sstevel@tonic-gate		gettext "                   a port defined (-v option) or\n" 1>&2
2467c478bd9Sstevel@tonic-gate		gettext "                   have dial-out instructions (-U option).\n" 1>&2
2477c478bd9Sstevel@tonic-gate		exit 1
2487c478bd9Sstevel@tonic-gate	fi
2497c478bd9Sstevel@tonic-gatefi
2507c478bd9Sstevel@tonic-gate
251355b4669Sjacobs#	process the "server" value
252355b4669Sjacobs#	It can be a hostname, UUCP form (server!queue), RCMD form(queue@server),
253355b4669Sjacobs#	or in URI form ({scheme}://{endpoint})
254355b4669Sjacobs#
255355b4669Sjacobscase "${server}" in
256355b4669Sjacobs	*://*)	# URI form
257355b4669Sjacobs		uri=${server}
258b51e021dSjacobs		rem_printer=$(expr "${server}" : ".*://.*/\([^/]*\)")
259b51e021dSjacobs		server=$(expr "${server}" : ".*://\([^/]*\)/.*")
260355b4669Sjacobs		;;
261355b4669Sjacobs	*@*)	# RCMD form
262b51e021dSjacobs		rem_printer=$(expr "${server}" : "\(.*\)@.*")
263b51e021dSjacobs		server=$(expr "${server}" : ".*@\(.*\)")
264355b4669Sjacobs		;;
265355b4669Sjacobs	*!*)	# UUCP form
266b51e021dSjacobs		rem_printer=$(expr "${server}" : ".*!\(.*\)")
267b51e021dSjacobs		server=$(expr "${server}" : "\(.*\)!.*")
268355b4669Sjacobs		;;
269355b4669Sjacobs	*)	# hostname
2707c478bd9Sstevel@tonic-gate		rem_printer=${printer}
271355b4669Sjacobs		;;
272355b4669Sjacobsesac
273b51e021dSjacobs
274b51e021dSjacobsif [[ -n "${server}" ]] ; then
275*0a44ef6dSjacobs	# if we need a uri, find the "best" one.
276*0a44ef6dSjacobs	if [[ -z "${uri}" ]] ; then
277*0a44ef6dSjacobs		uri="ipp://${server}/printers/${rem_printer}"
278*0a44ef6dSjacobs		${LPSTAT} -p ${uri} >/dev/null 2>&1
279*0a44ef6dSjacobs		if [[ $? -ne 0 ]] ; then
280*0a44ef6dSjacobs			uri="lpd://${server}/printers/${rem_printer}#Solaris"
281*0a44ef6dSjacobs		fi
282*0a44ef6dSjacobs	fi
283*0a44ef6dSjacobs	# set the bsdaddr
284355b4669Sjacobs	bsdaddr="${server},${rem_printer},Solaris"
2857c478bd9Sstevel@tonic-gatefi
2867c478bd9Sstevel@tonic-gate
287*0a44ef6dSjacobs# if there is a "device" or LP configuration, it's local
288*0a44ef6dSjacobsif [[ -n "${device}" || -f /etc/lp/printers/${printer}/configuration || \
289*0a44ef6dSjacobs      -f /etc/lp/classes/${printer} ]] ; then
290*0a44ef6dSjacobs	local="true"
291*0a44ef6dSjacobsfi
292b51e021dSjacobs
293b51e021dSjacobs# Do the LP configuration for a local printer served by lpsched
294*0a44ef6dSjacobsif [[ -x ${LPADMIN} && -n "${local}" ]] ; then
295b51e021dSjacobs	# enumerate LP configured printers before modification
296b51e021dSjacobs	PRE=$(/bin/ls /etc/lp/printers 2>/dev/null ; /bin/ls /etc/lp/classes \
297b51e021dSjacobs		2>/dev/null)
298b51e021dSjacobs
299b51e021dSjacobs	# if there are no printers configured, enable LP service(s)
300b51e021dSjacobs	[[ -z "${PRE}" ]] && lp_config_service enable
301b51e021dSjacobs
302b51e021dSjacobs	# add filters to LP service
303b51e021dSjacobs	lp_config_filters
304b51e021dSjacobs
305b51e021dSjacobs	# modify LP destination(s)
306b51e021dSjacobs	CMD=${LPADMIN}
307b51e021dSjacobs	while [[ -n "$*" ]] ; do	# to deal with multi-word arguments
308b51e021dSjacobs		CMD="$CMD \"$1\""
309b51e021dSjacobs		shift
310b51e021dSjacobs	done
311b51e021dSjacobs	case "$CMD" in
312b51e021dSjacobs		*\"-D\")
313b51e021dSjacobs			CMD="$CMD \"\""
314b51e021dSjacobs		;;
315b51e021dSjacobs	esac
316b51e021dSjacobs
317b51e021dSjacobs	# execute the LP lpadmin command
318b51e021dSjacobs	eval $CMD
319b51e021dSjacobs	exit_code=$?
320b51e021dSjacobs
321b51e021dSjacobs	# enumerate LP configured printers after modification
322b51e021dSjacobs	POST=$(/bin/ls /etc/lp/printers 2>/dev/null ; /bin/ls /etc/lp/classes \
323b51e021dSjacobs                    2>/dev/null)
324b51e021dSjacobs
325b51e021dSjacobs	# if there are no destinations, disable the service(s)
326b51e021dSjacobs	[[ -z "${POST}" ]] && lp_config_service disable
327b51e021dSjacobs
328b51e021dSjacobs	# sync printers.conf with LP configuration
329b51e021dSjacobs	lp_config_sync_pconf "${PRE}" "${POST}"
330b51e021dSjacobsfi
331b51e021dSjacobs
332b51e021dSjacobs# Do any printers.conf configuration that is required
333b51e021dSjacobsif [[ -n "${delete}" ]] ; then
334b51e021dSjacobs	if [[ "${delete}" = "all" ]] ; then
335b51e021dSjacobs		[[ $exit_code -eq 0 ]] && delete_all
336*0a44ef6dSjacobs   	elif [[ -z "${local}" ]] ; then
3377c478bd9Sstevel@tonic-gate   		${LPSET} -n system -x ${delete}
3387c478bd9Sstevel@tonic-gate   		exit_code=$?
3397c478bd9Sstevel@tonic-gate   	fi
340*0a44ef6dSjacobselif [[ -z "${local}" ]] ; then
341b51e021dSjacobs	if [[ -n "${printer}" ]] ; then
3427c478bd9Sstevel@tonic-gate		${LPSET} -n system \
343355b4669Sjacobs			-a "printer-uri-supported=${uri}" \
344b51e021dSjacobs			-a "bsdaddr=${bsdaddr}" ${printer}
345b51e021dSjacobs		exit_code=$?
346b51e021dSjacobs	fi
347b51e021dSjacobs
348b51e021dSjacobs	if [[ -n "${printer}" && -n "${description}" ]] ; then
349b51e021dSjacobs		${LPSET} -n system \
350b51e021dSjacobs			-a "description=${description}" ${printer}
3517c478bd9Sstevel@tonic-gate		exit_code=$?
3527c478bd9Sstevel@tonic-gate	fi
3537c478bd9Sstevel@tonic-gatefi
3547c478bd9Sstevel@tonic-gate
355b51e021dSjacobs# if the "default" doesn't resolve a "bsdaddr", the printer is gone, remove it
356b51e021dSjacobs${LPGET} -n system -k bsdaddr _default >/dev/null 2>&1 ||
357b51e021dSjacobs	${LPSET} -n system -x _default >/dev/null 2>&1
3587c478bd9Sstevel@tonic-gate
3597c478bd9Sstevel@tonic-gateexit $exit_code
360