xref: /csrg-svn/usr.sbin/sendmail/cf/m4/proto.m4 (revision 69629)
151222Sericdivert(-1)
251222Seric#
368840Seric# Copyright (c) 1983, 1995 Eric P. Allman
463592Sbostic# Copyright (c) 1988, 1993
563592Sbostic#	The Regents of the University of California.  All rights reserved.
651222Seric#
751222Seric# %sccs.include.redist.sh%
851222Seric#
951222Sericdivert(0)
1051222Seric
11*69629SericVERSIONID(`@(#)proto.m4	8.74 (Berkeley) 05/23/95')
1251222Seric
1351322SericMAILER(local)dnl
1451222Seric
1567696Seric# level 6 config file format
1668651SericV6/Berkeley
1768651Sericdivert(-1)
1868651Seric
1968830Seric# do some sanity checking
2068830Sericifdef(`__OSTYPE__',,
2168830Seric	`errprint(`*** ERROR: No system type defined (use OSTYPE macro)')')
2268830Seric
2368651Seric# pick our default mailers
2468651Sericifdef(`confSMTP_MAILER',, `define(`confSMTP_MAILER', `smtp')')
2568651Sericifdef(`confLOCAL_MAILER',, `define(`confLOCAL_MAILER', `local')')
2664010Sericifdef(`confRELAY_MAILER',,
2764010Seric	`define(`confRELAY_MAILER',
2864010Seric		`ifdef(`_MAILER_smtp_', `relay',
2968651Seric			`ifdef(`_MAILER_uucp', `suucp', `unknown')')')')
3063998Sericdefine(`_SMTP_', `confSMTP_MAILER')dnl		for readability only
3163998Sericdefine(`_LOCAL_', `confLOCAL_MAILER')dnl	for readability only
3264010Sericdefine(`_RELAY_', `confRELAY_MAILER')dnl	for readability only
3352646Seric
3468651Seric# back compatibility with old config files
3568651Sericifdef(`confDEF_GROUP_ID',
3668651Seric	`errprint(`*** confDEF_GROUP_ID is obsolete.')
3768651Seric	 errprint(`    Use confDEF_USER_ID with a colon in the value instead.')')
3868651Sericifdef(`confREAD_TIMEOUT',
3968651Seric	`errprint(`*** confREAD_TIMEOUT is obsolete.')
4068651Seric	 errprint(`    Use individual confTO_<timeout> parameters instead.')')
4168651Sericifdef(`confMESSAGE_TIMEOUT',
4268651Seric	`define(`_ARG_', index(confMESSAGE_TIMEOUT, /))
4368651Seric	 ifelse(_ARG_, -1,
4468651Seric		`define(`confTO_QUEUERETURN', confMESSAGE_TIMEOUT)',
4568651Seric		`define(`confTO_QUEUERETURN',
4668651Seric			substr(confMESSAGE_TIMEOUT, 0, _ARG_))
4768651Seric		 define(`confTO_QUEUEWARN',
4868651Seric			substr(confMESSAGE_TIMEOUT, eval(_ARG_+1)))')')
4968651Sericifdef(`confMIN_FREE_BLOCKS', `ifelse(index(confMIN_FREE_BLOCKS, /), -1,,
5068651Seric	`errprint(`*** compound confMIN_FREE_BLOCKS is obsolete.')
5168651Seric	 errprint(`    Use confMAX_MESSAGE_SIZE for the second part of the value.')')')
5268651Seric
5368651Seric# clean option definitions below....
5468651Sericdefine(`_OPTION', `ifdef(`$2', `O $1=$2', `#O $1`'ifelse($3, `',, `=$3')')')dnl
5568651Seric
5668651Sericdivert(0)dnl
5768651Seric
5851222Seric##################
5951222Seric#   local info   #
6051222Seric##################
6151222Seric
6251889SericCwlocalhost
6351887Sericifdef(`USE_CW_FILE',
6451889Seric`# file containing names of hosts for which we receive email
6564033SericFw`'confCW_FILE',
6664033Seric	`dnl')
6768651Seric
6864321Seric# my official domain name
6968651Seric# ... define this only if sendmail cannot automatically determine your domain
7068651Sericifdef(`confDOMAIN_NAME', `Dj`'confDOMAIN_NAME', `#Dj$w.Foo.COM')
7151222Seric
7268651Sericifdef(`_NULL_CLIENT_ONLY_', `divert(-1)')dnl
7364321Seric
7464321SericCP.
7564321Seric
7651222Sericifdef(`UUCP_RELAY',
7751222Seric`# UUCP relay host
7864033SericDY`'UUCP_RELAY
7960212SericCPUUCP
8057944Seric
8151222Seric')dnl
8251222Sericifdef(`BITNET_RELAY',
8351222Seric`#  BITNET relay host
8464033SericDB`'BITNET_RELAY
8560212SericCPBITNET
8657944Seric
8751222Seric')dnl
8869625Sericifdef(`DECNET_RELAY',
89*69629Seric`define(`_USE_DECNET_SYNTAX_')dnl
9069625Seric# DECnet relay host
9169625SericDC`'DECNET_RELAY
9269625SericCPDECNET
9369625Seric
9469625Seric')dnl
9558364Sericifdef(`FAX_RELAY',
9658364Seric`# FAX relay host
9764033SericDF`'FAX_RELAY
9860212SericCPFAX
9958364Seric
10058364Seric')dnl
10164321Seric# "Smart" relay host (may be null)
10264321SericDS`'ifdef(`SMART_HOST', SMART_HOST)
10351222Seric
10467914Sericifdef(`LUSER_RELAY',
10567914Seric`# place to which unknown users should be forwarded
10667914SericKuser user -m -a<>
10767914SericDL`'LUSER_RELAY
10867914Seric', `dnl')
10967914Seric
11068651Seric# operators that cannot be in local usernames (i.e., network indicators)
11168651SericCO @ % ifdef(`_NO_UUCP_', `', `!')
11268651Seric
11368651Seric# a class with just dot (for identifying canonical names)
11468651SericC..
11568651Seric
11658781Sericifdef(`MAILER_TABLE',
11758781Seric`# Mailer table (overriding domains)
11858781SericKmailertable MAILER_TABLE
11958781Seric
12058781Seric')dnl
12163754Sericifdef(`DOMAIN_TABLE',
12263754Seric`# Domain table (adding domains)
12363754SericKdomaintable DOMAIN_TABLE
12463754Seric
12563754Seric')dnl
12651222Seric# who I send unqualified names to (null means deliver locally)
12764033SericDR`'ifdef(`LOCAL_RELAY', LOCAL_RELAY)
12851222Seric
12957527Seric# who gets all local email traffic ($R has precedence for unqualified names)
13064033SericDH`'ifdef(`MAIL_HUB', MAIL_HUB)
13157527Seric
13257245Seric# class L: names that should be delivered locally, even if we have a relay
13357245Seric# class E: names that should be exposed as from this host, even if we masquerade
13467538Seric#CL root
13567538SericCE root
13651279Sericundivert(5)dnl
13751279Seric
13867469Seric# dequoting map
13967469SericKdequote dequote
14057944Seric
14168651Sericdivert(0)dnl
14268651Seric# who I masquerade as (null for no masquerading)
14368651SericDM`'ifdef(`MASQUERADE_NAME', MASQUERADE_NAME)
14468651Seric
14559033Sericundivert(6)dnl
14659033Seric
14757944Seric######################
14857944Seric#   Special macros   #
14957944Seric######################
15057944Seric
15157944Seric# SMTP initial login message
15264033SericDe`'confSMTP_LOGIN_MSG
15357944Seric
15457944Seric# UNIX initial From header format
15564033SericDl`'confFROM_LINE
15657944Seric
15757944Seric# my name for error messages
15864033SericDn`'confMAILER_NAME
15957944Seric
16057944Seric# delimiter (operator) characters
16164033SericDo`'confOPERATORS
16257944Seric
16357944Seric# format of a total name
16467538SericDq`'ifdef(`confFROM_HEADER', confFROM_HEADER, `$?x$x <$g>$|$g$.')
16557944Sericinclude(`../m4/version.m4')
16657944Seric
16757944Seric###############
16857944Seric#   Options   #
16957944Seric###############
17057944Seric
17159887Seric# strip message body to 7 bits on input?
17268651Seric_OPTION(SevenBitInput, `confSEVEN_BIT_INPUT')
17357944Seric
17467550Seric# 8-bit data handling
17568651Seric_OPTION(EightBitMode, `confEIGHT_BIT_HANDLING', adaptive)
17667550Seric
17768651Seric# wait for alias file rebuild (default units: minutes)
17868651Seric_OPTION(AliasWait, `confALIAS_WAIT', 5m)
17957944Seric
18057944Seric# location of alias file
18168651SericO AliasFile=ifdef(`ALIAS_FILE', `ALIAS_FILE', /etc/aliases)
18257944Seric
18358084Seric# minimum number of free blocks on filesystem
18468651Seric_OPTION(MinFreeBlocks, `confMIN_FREE_BLOCKS', 100)
18558084Seric
18668651Seric# maximum message size
18768651Seric_OPTION(MaxMessageSize, `confMAX_MESSAGE_SIZE', 1000000)
18868651Seric
18957944Seric# substitution for space (blank) characters
19068651Seric_OPTION(BlankSub, `confBLANK_SUB', _)
19157944Seric
19265618Seric# avoid connecting to "expensive" mailers on initial submission?
19368651Seric_OPTION(HoldExpensive, `confCON_EXPENSIVE')
19457944Seric
19557944Seric# checkpoint queue runs after every N successful deliveries
19668651Seric_OPTION(CheckpointInterval, `confCHECKPOINT_INTERVAL', 10)
19757944Seric
19857944Seric# default delivery mode
19968651Seric_OPTION(DeliveryMode, `confDELIVERY_MODE', background)
20057944Seric
20157944Seric# automatically rebuild the alias database?
20268651Seric_OPTION(AutoRebuildAliases, `confAUTO_REBUILD')
20357944Seric
20464033Seric# error message header/file
20568651Seric_OPTION(ErrorHeader, `confERROR_MESSAGE', /etc/sendmail.oE)
20657944Seric
20757944Seric# error mode
20868651Seric_OPTION(ErrorMode, `confERROR_MODE', print)
20957944Seric
21057944Seric# save Unix-style "From_" lines at top of header?
21168651Seric_OPTION(SaveFromLine, `confSAVE_FROM_LINES')
21257944Seric
21357944Seric# temporary file mode
21468651Seric_OPTION(TempFileMode, `confTEMP_FILE_MODE', 0600)
21557944Seric
21657944Seric# match recipients against GECOS field?
21768651Seric_OPTION(MatchGECOS, `confMATCH_GECOS')
21857944Seric
21957944Seric# maximum hop count
22068651Seric_OPTION(MaxHopCount, `confMAX_HOP', 17)
22157944Seric
22257944Seric# location of help file
22368651SericO HelpFile=ifdef(`HELP_FILE', HELP_FILE, /usr/lib/sendmail.hf)
22457944Seric
22557944Seric# ignore dots as terminators in incoming messages?
22668651Seric_OPTION(IgnoreDots, `confIGNORE_DOTS')
22757944Seric
22868651Seric# name resolver options
22968651Seric_OPTION(ResolverOptions, `confBIND_OPTS', +AAONLY)
23057944Seric
23159887Seric# deliver MIME-encapsulated error messages?
23268651Seric_OPTION(SendMimeErrors, `confMIME_FORMAT_ERRORS')
23359887Seric
23457944Seric# Forward file search path
23568651Seric_OPTION(ForwardPath, `confFORWARD_PATH', /var/forward/$u:$z/.forward.$w:$z/.forward)
23657944Seric
23757944Seric# open connection cache size
23868651Seric_OPTION(ConnectionCacheSize, `confMCI_CACHE_SIZE', 2)
23957944Seric
24057944Seric# open connection cache timeout
24168651Seric_OPTION(ConnectionCacheTimeout, `confMCI_CACHE_TIMEOUT', 5m)
24257944Seric
24361106Seric# use Errors-To: header?
24468651Seric_OPTION(UseErrorsTo, `confUSE_ERRORS_TO')
24561106Seric
24657944Seric# log level
24768651Seric_OPTION(LogLevel, `confLOG_LEVEL', 10)
24857944Seric
24957944Seric# send to me too, even in an alias expansion?
25068651Seric_OPTION(MeToo, `confME_TOO')
25157944Seric
25257944Seric# verify RHS in newaliases?
25368651Seric_OPTION(CheckAliases, `confCHECK_ALIASES')
25457944Seric
25557944Seric# default messages to old style headers if no special punctuation?
25668651Seric_OPTION(OldStyleHeaders, `confOLD_STYLE_HEADERS')
25757944Seric
25858860Seric# SMTP daemon options
25968651Seric_OPTION(DaemonPortOptions, `confDAEMON_OPTIONS', Port=esmtp)
26058860Seric
26158084Seric# privacy flags
26268651Seric_OPTION(PrivacyOptions, `confPRIVACY_FLAGS', authwarnings)
26358084Seric
26457944Seric# who (if anyone) should get extra copies of error messages
26568651Seric_OPTION(PostMasterCopy, `confCOPY_ERRORS_TO', Postmaster)
26657944Seric
26757944Seric# slope of queue-only function
26868651Seric_OPTION(QueueFactor, `confQUEUE_FACTOR', 600000)
26957944Seric
27057944Seric# queue directory
27168651SericO QueueDirectory=ifdef(`QUEUE_DIR', QUEUE_DIR, /var/spool/mqueue)
27257944Seric
27368651Seric# timeouts (many of these)
27468651Seric_OPTION(Timeout.initial, `confTO_INITIAL', 5m)
27568651Seric_OPTION(Timeout.helo, `confTO_HELO', 5m)
27668651Seric_OPTION(Timeout.mail, `confTO_MAIL', 10m)
27768651Seric_OPTION(Timeout.rcpt, `confTO_RCPT', 1h)
27868651Seric_OPTION(Timeout.datainit, `confTO_DATAINIT', 5m)
27968651Seric_OPTION(Timeout.datablock, `confTO_DATABLOCK', 1h)
28068651Seric_OPTION(Timeout.datafinal, `confTO_DATAFINAL', 1h)
28168651Seric_OPTION(Timeout.rset, `confTO_RSET', 5m)
28268651Seric_OPTION(Timeout.quit, `confTO_QUIT', 2m)
28368651Seric_OPTION(Timeout.misc, `confTO_MISC', 2m)
28468651Seric_OPTION(Timeout.command, `confTO_COMMAND', 1h)
28568651Seric_OPTION(Timeout.ident, `confTO_IDENT', 30s)
28668651Seric_OPTION(Timeout.fileopen, `confTO_FILEOPEN', 60s)
28768651Seric_OPTION(Timeout.queuereturn, `confTO_QUEUERETURN', 5d)
28868651Seric_OPTION(Timeout.queuereturn.normal, `confTO_QUEUERETURN_NORMAL', 5d)
28968651Seric_OPTION(Timeout.queuereturn.urgent, `confTO_QUEUERETURN_URGENT', 2d)
29068651Seric_OPTION(Timeout.queuereturn.non-urgent, `confTO_QUEUERETURN_NONURGENT', 7d)
29168651Seric_OPTION(Timeout.queuewarn, `confTO_QUEUEWARN', 4h)
29268651Seric_OPTION(Timeout.queuewarn.normal, `confTO_QUEUEWARN_NORMAL', 4h)
29368651Seric_OPTION(Timeout.queuewarn.urgent, `confTO_QUEUEWARN_URGENT', 1h)
29468651Seric_OPTION(Timeout.queuewarn.non-urgent, `confTO_QUEUEWARN_NONURGENT', 12h)
29557944Seric
29667810Seric# should we not prune routes in route-addr syntax addresses?
29768651Seric_OPTION(DontPruneRoutes, `confDONT_PRUNE_ROUTES')
29867810Seric
29957944Seric# queue up everything before forking?
30068651Seric_OPTION(SuperSafe, `confSAFE_QUEUE')
30157944Seric
30257944Seric# status file
30368651Seric_OPTION(StatusFile, `STATUS_FILE', /etc/sendmail.st)
30457944Seric
30557944Seric# time zone handling:
30657944Seric#  if undefined, use system default
30757944Seric#  if defined but null, use TZ envariable passed in
30857944Seric#  if defined and non-null, use that info
30968651Sericifelse(confTIME_ZONE, `USE_SYSTEM', `#O TimeZoneSpec=',
31068651Seric	confTIME_ZONE, `USE_TZ', `O TimeZoneSpec=',
31168651Seric	`O TimeZoneSpec=confTIME_ZONE')
31257944Seric
31368651Seric# default UID (can be username or userid:groupid)
31468651Seric_OPTION(DefaultUser, `confDEF_USER_ID', nobody)
31557944Seric
31651887Seric# list of locations of user database file (null means no lookup)
31768651Seric_OPTION(UserDatabaseSpec, `confUSERDB_SPEC', /etc/userdb)
31851222Seric
31958860Seric# fallback MX host
32068651Seric_OPTION(FallbackMXhost, `confFALLBACK_MX', fall.back.host.net)
32158860Seric
32263858Seric# if we are the best MX host for a site, try it directly instead of config err
32368651Seric_OPTION(TryNullMXList, `confTRY_NULL_MX_LIST')
32463858Seric
32557944Seric# load average at which we just queue messages
32668651Seric_OPTION(QueueLA, `confQUEUE_LA', 8)
32757944Seric
32857944Seric# load average at which we refuse connections
32968651Seric_OPTION(RefuseLA, `confREFUSE_LA', 12)
33057944Seric
33157944Seric# work recipient factor
33268651Seric_OPTION(RecipientFactor, `confWORK_RECIPIENT_FACTOR', 30000)
33357944Seric
33457944Seric# deliver each queued job in a separate process?
33568651Seric_OPTION(ForkEachJob, `confSEPARATE_PROC')
33657944Seric
33757944Seric# work class factor
33868651Seric_OPTION(ClassFactor, `confWORK_CLASS_FACTOR', 1800)
33957944Seric
34057944Seric# work time factor
34168651Seric_OPTION(RetryFactor, `confWORK_TIME_FACTOR', 90000)
34257944Seric
34367697Seric# shall we sort the queue by hostname first?
34468651Seric_OPTION(QueueSortOrder, `confQUEUE_SORT_ORDER', priority)
34567697Seric
34668651Seric# minimum time in queue before retry
34768651Seric_OPTION(MinQueueAge, `confMIN_QUEUE_AGE', 30m)
34868651Seric
34968651Seric# default character set
35068651Seric_OPTION(DefaultCharSet, `confDEF_CHAR_SET', iso-8859-1)
35168651Seric
35268651Seric# service switch file (ignored on Solaris, Ultrix, OSF/1, others)
35368651Seric_OPTION(ServiceSwitchFile, `confSERVICE_SWITCH_FILE', /etc/service.switch)
35468651Seric
35568651Seric# dialup line delay on connection failure
35668651Seric_OPTION(DialDelay, `confDIAL_DELAY', 10s)
35768651Seric
35868651Seric# action to take if there are no recipients in the message
35968651Seric_OPTION(NoRecipientAction, `confNO_RCPT_ACTION', add-to-undisclosed)
36068651Seric
36168651Seric# chrooted environment for writing to files
36268651Seric_OPTION(SafeFileEnvironment, `confSAFE_FILE_ENV', /arch)
36368651Seric
36468806Seric# are colons OK in addresses?
36568806Seric_OPTION(ColonOkInAddr, `confCOLON_OK_IN_ADDR')
36668806Seric
36757944Seric###########################
36857944Seric#   Message precedences   #
36957944Seric###########################
37057944Seric
37157944SericPfirst-class=0
37257944SericPspecial-delivery=100
37358070SericPlist=-30
37457944SericPbulk=-60
37557944SericPjunk=-100
37657944Seric
37757944Seric#####################
37857944Seric#   Trusted users   #
37957944Seric#####################
38057944Seric
38168651Seric# this is equivalent to setting class "t"
38268651Seric#Ft/etc/sendmail.trusted
38357944SericTroot
38457944SericTdaemon
38557944SericTuucp
38657944Seric
38757944Seric#########################
38857944Seric#   Format of headers   #
38957944Seric#########################
39057944Seric
39158679SericH?P?Return-Path: $g
39267819SericHReceived: confRECEIVED_HEADER
39357944SericH?D?Resent-Date: $a
39457944SericH?D?Date: $a
39557944SericH?F?Resent-From: $q
39657944SericH?F?From: $q
39757944SericH?x?Full-Name: $x
39857944SericHSubject:
39957944Seric# HPosted-Date: $a
40057944Seric# H?l?Received-Date: $b
40157944SericH?M?Resent-Message-Id: <$t.$i@$j>
40257944SericH?M?Message-Id: <$t.$i@$j>
40368651Sericifdef(`_NULL_CLIENT_ONLY_',
40468651Seric	`include(../m4/nullrelay.m4)m4exit',
40568651Seric	`dnl')
40651222Seric#
40751222Seric######################################################################
40851222Seric######################################################################
40951222Seric#####
41051222Seric#####			REWRITING RULES
41151222Seric#####
41251222Seric######################################################################
41351222Seric######################################################################
41451222Seric
41558678Sericundivert(9)dnl
41651222Seric
41751222Seric###########################################
41851222Seric###  Rulset 3 -- Name Canonicalization  ###
41951222Seric###########################################
42051260SericS3
42151222Seric
42264802Seric# handle null input (translate to <@> special case)
42358499SericR$@			$@ <@>
42458174Seric
42568763Seric# strip group: syntax (not inside angle brackets!) and trailing semicolon
42668763SericR$*			$: $1 <@>			mark addresses
42768763SericR$* < $* > $* <@>	$: $1 < $2 > $3			unmark <addr>
42869625SericR$* :: $* <@>		$: $1 :: $2			unmark node::addr
42969215SericR:include: $* <@>	$: :include: $1			unmark :include:...
43068763SericR$* : $* <@>		$: $2				strip colon if marked
43168763SericR$* <@>			$: $1				unmark
43268763SericR$* ;			$: $1				strip trailing semi
43368763Seric
43468763Seric# null input now results from list:; syntax
43568763SericR$@			$@ :; <@>
43668763Seric
43751222Seric# basic textual canonicalization -- note RFC733 heuristic here
43864144SericR$*<$*>$*<$*>$*		$2$3<$4>$5			strip multiple <> <>
43958158SericR$*<$*<$+>$*>$*		<$3>$5				2-level <> nesting
44058499SericR$*<>$*			$@ <@>				MAIL FROM:<> case
44151222SericR$*<$+>$*		$2				basic RFC821/822 parsing
44251222Seric
44351222Seric# make sure <@a,@b,@c:user@d> syntax is easy to parse -- undone later
44455416SericR@ $+ , $+		@ $1 : $2			change all "," to ":"
44551222Seric
44651222Seric# localize and dispose of route-based addresses
44767469SericR@ $+ : $+		$@ $>96 < @$1 > : $2		handle <route-addr>
44851222Seric
44951222Seric# find focus for list syntax
45067469SericR $+ : $* ; @ $+	$@ $>96 $1 : $2 ; < @ $3 >	list syntax
45155416SericR $+ : $* ;		$@ $1 : $2;			list syntax
45251222Seric
45351222Seric# find focus for @ syntax addresses
45455416SericR$+ @ $+		$: $1 < @ $2 >			focus on domain
45555416SericR$+ < $+ @ $+ >		$1 $2 < @ $3 >			move gaze right
45667469SericR$+ < @ $+ >		$@ $>96 $1 < @ $2 >		already canonical
45751222Seric
45864968Seric# do some sanity checking
45964968SericR$* < @ $* : $* > $*	$1 < @ $2 $3 > $4		nix colons in addrs
46064968Seric
46158288Sericifdef(`_NO_UUCP_', `dnl',
46258288Seric`# convert old-style addresses to a domain-based address
46367469SericR$- ! $+		$@ $>96 $2 < @ $1 .UUCP >	resolve uucp names
46467469SericR$+ . $- ! $+		$@ $>96 $3 < @ $1 . $2 >		domain uucps
46569625SericR$+ ! $+		$@ $>96 $2 < @ $1 .UUCP >	uucp subdomains
46669625Seric')
467*69629Sericifdef(`_USE_DECNET_SYNTAX_',
46869625Seric`# convert node::user addresses into a domain-based address
46969625SericR$- :: $+		$@ $>96 $2 < @ $1 .DECNET >	resolve DECnet names
470*69629SericR$- . $- :: $+		$@ $>96 $3 < @ $1.$2 .DECNET >	numeric DECnet addr
47169625Seric',
47269625Seric	`dnl')
47351222Seric# if we have % signs, take the rightmost one
47455416SericR$* % $*		$1 @ $2				First make them all @s.
47555416SericR$* @ $* @ $*		$1 % $2 @ $3			Undo all but the last.
47667469SericR$* @ $*		$@ $>96 $1 < @ $2 >		Insert < > and finish
47751222Seric
47851222Seric# else we must be a local name
47969542SericR$*			$@ $>96 $1
48051222Seric
48151222Seric
48260894Seric################################################
48367469Seric###  Ruleset 96 -- bottom half of ruleset 3  ###
48460894Seric################################################
48551222Seric
48667469SericS96
48751222Seric
48851222Seric# handle special cases for local names
48960262SericR$* < @ localhost > $*		$: $1 < @ $j . > $2		no domain at all
49060262SericR$* < @ localhost . $m > $*	$: $1 < @ $j . > $2		local domain
49160262Sericifdef(`_NO_UUCP_', `dnl',
49260262Seric`R$* < @ localhost . UUCP > $*	$: $1 < @ $j . > $2		.UUCP domain')
49365825SericR$* < @ [ $+ ] > $*		$: $1 < @@ [ $2 ] > $3		mark [a.b.c.d]
49465825SericR$* < @@ $=w > $*		$: $1 < @ $j . > $3		self-literal
49566306SericR$* < @@ $+ > $*		$@ $1 < @ $2 > $3		canon IP addr
49663754Sericifdef(`DOMAIN_TABLE', `
49767450Seric# look up domains in the domain table
49867450SericR$* < @ $+ > $*			$: $1 < @ $(domaintable $2 $) > $3',
49963754Seric`dnl')
50051889Sericundivert(2)dnl
50151222Seric
50259081Sericifdef(`_NO_UUCP_', `dnl',
50359081Seric`ifdef(`UUCP_RELAY',
50451274Seric`# pass UUCP addresses straight through
50564971SericR$* < @ $+ . UUCP > $*		$@ $1 < @ $2 . UUCP . > $3',
50651274Seric`# if really UUCP, handle it immediately
50751322Sericifdef(`_CLASS_U_',
50864971Seric`R$* < @ $=U . UUCP > $*	$@ $1 < @ $2 . UUCP . > $3', `dnl')
50951322Sericifdef(`_CLASS_V_',
51064971Seric`R$* < @ $=V . UUCP > $*	$@ $1 < @ $2 . UUCP . > $3', `dnl')
51151322Sericifdef(`_CLASS_W_',
51264971Seric`R$* < @ $=W . UUCP > $*	$@ $1 < @ $2 . UUCP . > $3', `dnl')
51351322Sericifdef(`_CLASS_X_',
51464971Seric`R$* < @ $=X . UUCP > $*	$@ $1 < @ $2 . UUCP . > $3', `dnl')
51551322Sericifdef(`_CLASS_Y_',
51664971Seric`R$* < @ $=Y . UUCP > $*	$@ $1 < @ $2 . UUCP . > $3', `dnl')
51751267Seric
51851267Seric# try UUCP traffic as a local address
51964971SericR$* < @ $+ . UUCP > $*		$: $1 < @ $[ $2 $] . UUCP . > $3
52067469SericR$* < @ $+ . . UUCP . > $*		$@ $1 < @ $2 . > $3')
52159081Seric')
52265061Sericifdef(`_NO_CANONIFY_', `dnl',
52359081Seric`# pass to name server to make hostname canonical
52461470SericR$* < @ $* $~P > $*		$: $1 < @ $[ $2 $3 $] > $4')
52561470Seric
52665061Seric# local host aliases and pseudo-domains are always canonical
52765061SericR$* < @ $=w > $*		$: $1 < @ $2 . > $3
52864971SericR$* < @ $* $=P > $*		$: $1 < @ $2 $3 . > $4
52965061SericR$* < @ $* . . > $*		$1 < @ $2 . > $3
53064971Seric
53151322Seric# if this is the local hostname, make sure we treat is as canonical
53251322SericR$* < @ $j > $*			$: $1 < @ $j . > $2
53351222Seric
53451322Seric
53551222Seric##################################################
53651222Seric###  Ruleset 4 -- Final Output Post-rewriting  ###
53751222Seric##################################################
53851222SericS4
53951222Seric
54069215SericR$* <@>			$@				handle <> and list:;
54151222Seric
54251322Seric# strip trailing dot off possibly canonical name
54351322SericR$* < @ $+ . > $*	$1 < @ $2 > $3
54451322Seric
54551222Seric# externalize local domain info
54651322SericR$* < $+ > $*		$1 $2 $3			defocus
54751322SericR@ $+ : @ $+ : $+	@ $1 , @ $2 : $3		<route-addr> canonical
54851997SericR@ $*			$@ @ $1				... and exit
54951222Seric
55058288Sericifdef(`_NO_UUCP_', `dnl',
55158288Seric`# UUCP must always be presented in old form
55258288SericR$+ @ $- . UUCP		$2!$1				u@h.UUCP => h!u')
55351222Seric
554*69629Sericifdef(`_USE_DECNET_SYNTAX_',
555*69629Seric`# put DECnet back in :: form
556*69629SericR$+ @ $+ . DECNET	$2 :: $1			u@h.DECNET => h::u',
557*69629Seric	`dnl')
55851222Seric# delete duplicate local names
55951322SericR$+ % $=w @ $=w		$1 @ $j				u%host@host => u@host
56051222Seric
56151222Seric
56251222Seric
56360894Seric##############################################################
56467469Seric###   Ruleset 97 -- recanonicalize and call ruleset zero   ###
56560894Seric###		   (used for recursive calls)		   ###
56660894Seric##############################################################
56751222Seric
56867469SericS`'97
56951222SericR$*			$: $>3 $1
57051222SericR$*			$@ $>0 $1
57151222Seric
57251222Seric
57351222Seric######################################
57451222Seric###   Ruleset 0 -- Parse Address   ###
57551222Seric######################################
57651222Seric
57751222SericS0
57851222Seric
57967118SericR<@>			$#_LOCAL_ $: <@>		special case error msgs
58068852SericR$* : $* ; <@>		$#error $@ 5.1.3 $: "list:; syntax illegal for recipient addresses"
58168852SericR<@ $+>			$#error $@ 5.1.1 $: "user address required"
58268852SericR$* <$* : $* > $*	$#error $@ 5.1.1 $: "colon illegal in host name part"
58368852SericR$* < @ . > $*		$#error $@ 5.1.2 $: "invalid host name"
58458340Seric
58555416Sericifdef(`_MAILER_smtp_',
58655416Seric`# handle numeric address spec
58767469SericR$* < @ [ $+ ] > $*	$: $>98 $1 < @ [ $2 ] > $3	numeric internet spec
58865191SericR$* < @ [ $+ ] > $*	$#_SMTP_ $@ [$2] $: $1 < @ [$2] > $3	still numeric: send',
58963858Seric	`dnl')
59051222Seric
59151887Seric# now delete the local info -- note $=O to find characters that cause forwarding
59267469SericR$* < @ > $*		$@ $>97 $1		user@ => user
59367469SericR< @ $=w . > : $*	$@ $>97 $2		@here:... -> ...
59467469SericR$* $=O $* < @ $=w . >	$@ $>97 $1 $2 $3		...@here -> ...
59563967Seric
59663967Seric# handle local hacks
59767469SericR$*			$: $>98 $1
59866048Seric
59966048Seric# short circuit local delivery so forwarded email works
60067914Sericifdef(`_STICKY_LOCAL_DOMAIN_',
60167914Seric`R$+ < @ $=w . >		$: < $H > $1 < @ $2 . >		first try hub
60268698SericR< $+ > $+ < $+ >	$>95 < $1 > $2 < $3 >		yep ....
60368155SericR< > $+ + $* < $+ >	$#_LOCAL_ $: $1 + $2		plussed name?
60467914SericR< > $+ < $+ >		$#_LOCAL_ $: @ $1			nope, local address',
60568698Seric`R$=L < @ $=w . >	$#_LOCAL_ $: @ $1		special local names
60668155SericR$+ < @ $=w . >		$#_LOCAL_ $: $1			regular local name')
60766090Sericifdef(`MAILER_TABLE',
60866090Seric`
60966090Seric# not local -- try mailer table lookup
61066090SericR$* <@ $+ > $*		$: < $2 > $1 < @ $2 > $3	extract host name
61166090SericR< $+ . > $*		$: < $1 > $2			strip trailing dot
61266090SericR< $+ > $*		$: < $(mailertable $1 $) > $2	lookup
61366090SericR< $- : $+ > $*		$# $1 $@ $2 $: $3		check -- resolved?
61466090SericR< $+ > $*		$: $>90 <$1> $2			try domain',
61566090Seric`dnl')
61651260Sericundivert(4)dnl
61751260Seric
61861468Sericifdef(`_NO_UUCP_', `dnl',
61961468Seric`# resolve remotely connected UUCP links (if any)
62051285Sericifdef(`_CLASS_V_',
62167469Seric`R$* < @ $=V . UUCP . > $*		$: $>95 < $V > $1 <@$2.UUCP.> $3',
62252044Seric	`dnl')
62351285Sericifdef(`_CLASS_W_',
62467469Seric`R$* < @ $=W . UUCP . > $*		$: $>95 < $W > $1 <@$2.UUCP.> $3',
62552044Seric	`dnl')
62651285Sericifdef(`_CLASS_X_',
62767469Seric`R$* < @ $=X . UUCP . > $*		$: $>95 < $X > $1 <@$2.UUCP.> $3',
62861468Seric	`dnl')')
62951260Seric
63051222Seric# resolve fake top level domains by forwarding to other hosts
63151222Sericifdef(`BITNET_RELAY',
63267469Seric`R$*<@$+.BITNET.>$*	$: $>95 < $B > $1 <@$2.BITNET.> $3	user@host.BITNET',
63351222Seric	`dnl')
63469625Sericifdef(`DECNET_RELAY',
63569625Seric`R$*<@$+.DECNET.>$*	$: $>95 < $C > $1 <@$2.DECNET.> $3	user@host.DECNET',
63669625Seric	`dnl')
63765149Sericifdef(`_MAILER_pop_',
63865149Seric`R$+ < @ POP. >		$#pop $: $1			user@POP',
63965149Seric	`dnl')
64058364Sericifdef(`_MAILER_fax_',
64164971Seric`R$+ < @ $+ .FAX. >	$#fax $@ $2 $: $1		user@host.FAX',
64258364Seric`ifdef(`FAX_RELAY',
64367469Seric`R$*<@$+.FAX.>$*		$: $>95 < $F > $1 <@$2.FAX.> $3	user@host.FAX',
64458364Seric	`dnl')')
64551222Seric
64651260Sericifdef(`UUCP_RELAY',
64751260Seric`# forward non-local UUCP traffic to our UUCP relay
64867469SericR$*<@$*.UUCP.>$*		$: $>95 < $Y > $1 <@$2.UUCP.> $3	uucp mail',
64955416Seric`ifdef(`_MAILER_uucp_',
65051260Seric`# forward other UUCP traffic straight to UUCP
65165557SericR$* < @ $+ .UUCP. > $*		$#uucp $@ $2 $: $1 < @ $2 .UUCP. > $3	user@host.UUCP',
65251260Seric	`dnl')')
65363967Sericifdef(`_MAILER_usenet_', `
65463754Seric# addresses sent to net.group.USENET will get forwarded to a newsgroup
65565851SericR$+ . USENET		$#usenet $: $1',
65663860Seric	`dnl')
65763754Seric
65858070Sericifdef(`_LOCAL_RULES_',
65958070Seric`# figure out what should stay in our local mail system
66063754Sericundivert(1)', `dnl')
66163754Seric
66264321Seric# pass names that still have a host to a smarthost (if defined)
66367469SericR$* < @ $* > $*		$: $>95 < $S > $1 < @ $2 > $3	glue on smarthost name
66463754Seric
66564321Seric# deal with other remote names
66663946Sericifdef(`_MAILER_smtp_',
66763973Seric`R$* < @$* > $*		$#_SMTP_ $@ $2 $: $1 < @ $2 > $3		user@host.domain',
66868852Seric`R$* < @$* > $*		$#error $@ 5.1.2 $: Unrecognized host name $2')
66951222Seric
67067469Seric# if this is quoted, strip the quotes and try again
67158811SericR$+			$: $(dequote $1 $)		strip quotes
67267469SericR$+ $=O $+		$@ $>97 $1 $2 $3			try again
67358811Seric
67458811Seric# handle locally delivered names
67563998SericR$=L			$#_LOCAL_ $: @ $1			special local names
67663998SericR$+			$#_LOCAL_ $: $1			regular local names
67751222Seric
67855416Seric###########################################################################
67955416Seric###   Ruleset 5 -- special rewriting after aliases have been expanded   ###
68055416Seric###########################################################################
68151222Seric
68251322SericS5
68351322Seric
68469542Seric# deal with plussed users so aliases work nicely
68569542SericR$+ + *			$#_LOCAL_ $@ $&h $: $1
68669542SericR$+ + $*		$#_LOCAL_ $@ $2 $: $1 + *
68767538Seric
68867914Seric# prepend an empty "forward host" on the front
68967914SericR$+			$: <> $1
69067914Seric
69167914Sericifdef(`LUSER_RELAY',
69267914Seric`# send unrecognized local users to a relay host
69368149SericR< > $+ + $*		$: < $L . > $( user $1 $) + $2
69468149SericR< > $+			$: < $L . > $( user $1 $)	look up user
69568149SericR< $* > $+ <> $*	$: < > $2 $3			found; strip $L
69668149SericR< $* . > $+		$: < $1 > $2			strip extra dot')
69767914Seric
69868149Seric# handle plussed local names
69968149SericR< > $+ + $*		$#_LOCAL_ $@ $2 $: $1
70068149Seric
70158808Seric# see if we have a relay or a hub
70268698SericR< > $+			$: < $H > $1			try hub
70367914SericR< > $+			$: < $R > $1			try relay
70466048SericR< > $+			$@ $1				nope, give up
70567469SericR< $- : $+ > $+		$: $>95 < $1 : $2 > $3 < @ $2 >
70667469SericR< $+ > $+		$@ $>95 < $1 > $2 < @ $1 >
70760540Sericifdef(`MAILER_TABLE',
70860540Seric`
70960540Seric
71063858Seric###################################################################
71163858Seric###  Ruleset 90 -- try domain part of mailertable entry 	###
71263858Seric###################################################################
71360540Seric
71460540SericS90
71566091SericR$* <$- . $+ > $*	$: $1$2 < $(mailertable .$3 $@ $1$2 $@ $2 $) > $4
71666081SericR$* <$- : $+ > $*	$# $2 $@ $3 $: $4		check -- resolved?
71766081SericR$* < . $+ > $*		$@ $>90 $1 . <$2> $3		no -- strip & try again
71866094SericR$* < $* > $*		$: < $(mailertable . $@ $1$2 $) > $3	try "."
71966090SericR<$- : $+ > $*		$# $1 $@ $2 $: $3		"." found?
72066090SericR< $* > $*		$@ $2				no mailertable match',
72160540Seric`dnl')
72263858Seric
72363858Seric###################################################################
72467469Seric###  Ruleset 95 -- canonify mailer:host syntax to triple	###
72564010Seric###################################################################
72664010Seric
72767469SericS95
72864010SericR< > $*			$@ $1				strip off null relay
72964010SericR< $- : $+ > $*		$# $1 $@ $2 $: $3		try qualified mailer
73064010SericR< $=w > $*		$@ $2				delete local host
73164010SericR< $+ > $*		$#_RELAY_ $@ $1 $: $2		use unqualified mailer
73264010Seric
73364010Seric###################################################################
73467469Seric###  Ruleset 98 -- local part of ruleset zero (can be null)	###
73563858Seric###################################################################
73663858Seric
73767469SericS98
73863858Sericundivert(3)dnl
73951222Seric#
74051222Seric######################################################################
74151222Seric######################################################################
74251222Seric#####
74356790Seric`#####			MAILER DEFINITIONS'
74451222Seric#####
74551222Seric######################################################################
74651222Seric######################################################################
74751222Sericundivert(7)dnl
748