diff --git a/postfix-3.7.0-whitespace-name-fix.patch b/postfix-3.7.0-whitespace-name-fix.patch deleted file mode 100644 index e6a1db438e13cc23328acba7482be0f4a99dcc69..0000000000000000000000000000000000000000 --- a/postfix-3.7.0-whitespace-name-fix.patch +++ /dev/null @@ -1,13 +0,0 @@ -diff --git a/src/cleanup/cleanup_message.c b/src/cleanup/cleanup_message.c -index f2b86d0..018e39e 100644 ---- a/src/cleanup/cleanup_message.c -+++ b/src/cleanup/cleanup_message.c -@@ -772,6 +772,8 @@ static void cleanup_header_done_callback(void *context) - /* Normalize whitespace. */ - token = tok822_scan_limit(state->fullname, &dummy_token, - var_token_limit); -+ if (!token) -+ token = tok822_alloc(TOK822_QSTRING, state->fullname); - } else { - token = tok822_alloc(TOK822_QSTRING, state->fullname); - } diff --git a/postfix-3.7.9-SRV-resolve.patch b/postfix-3.7.9-SRV-resolve.patch deleted file mode 100644 index 48171c34fe433d601479c3177fba6ad30000375a..0000000000000000000000000000000000000000 --- a/postfix-3.7.9-SRV-resolve.patch +++ /dev/null @@ -1,1584 +0,0 @@ -diff --git a/mantools/postlink b/mantools/postlink -index 15306d9..70ff038 100755 ---- a/mantools/postlink -+++ b/mantools/postlink -@@ -1161,6 +1161,10 @@ while (<>) { - s;\blocal_login_sender_maps\b;$&;g; - s;\bempty_address_local_login_sender_maps_lookup_key\b;$&;g; - -+ s;\buse_srv_lookup\b;$&;g; -+ s;\ballow_srv_lookup_fallback\b;$&;g; -+ s;\bignore_srv_lookup_error\b;$&;g; -+ - # Service-defined parameters... - - s;\bpolicy_time_limit\b;$&;g; -diff --git a/proto/postconf.proto b/proto/postconf.proto -index 787e01c..6c214e7 100644 ---- a/proto/postconf.proto -+++ b/proto/postconf.proto -@@ -18599,3 +18599,110 @@ to enable. This feature is enabled by default with Postfix ≥ - -

This feature is available in Postfix ≥ 3.9, 3.8.1, 3.7.6, - 3.6.10, and 3.5.20.

-+ -+%PARAM use_srv_lookup -+ -+

Enables discovery for the specified service(s) using DNS SRV -+records. For example, with "use_srv_lookup = submission" and -+"relayhost = example.com:submission", the Postfix SMTP client will -+look up DNS SRV records for _submission._tcp.example.com, and will -+relay email through the hosts and ports that are specified with -+those records. See RFC 2782 for details of the host selection -+process.

-+ -+

Specify zero or more service names separated by comma and/or -+whitespace. Any name in the services(5) database may be specified, -+though in practice only submission, submissions, and smtp make -+sense.

-+ -+

When SRV record lookup is enabled with use_srv_lookup, you can -+enclose a domain name in "[]" to force IP address lookup instead -+of SRV record lookup.

-+ -+

Example 1: MUA-to-MTA submission using SRV record lookup for -+the "submission" service for domain "example.com". This uses the -+default SMTP delivery agent with STARTTLS, and looks up SRV records -+for "_submission._tcp.example.com".

-+ -+
-+/etc/postfix/main.cf:
-+    use_srv_lookup = submission
-+    relayhost = example.com:submission
-+    smtp_tls_security_level = may
-+    ...see SASL_README for sasl configuration...
-+
-+ -+

Example 2: MUA-to-MTA submission using SRV record lookup for -+the "submissions" service for domain "example.org". This uses a -+dedicated SMTP delivery agent (smtp-wraptls) with tls_wrappermode -+turned on, and looks up SRV records for "_submissions._tcp.example.org". -+

-+ -+

Note: specify the older name "smtps" instead of "submissions" -+when a provider has DNS SRV records like "_smtps._tcp.example.org" -+instead of "_submissions._tcp.example.org".

-+ -+
-+/etc/postfix/main.cf:
-+    use_srv_lookup = submissions
-+    default_transport = smtp-wraptls:example.org:submissions
-+    ...see SASL_README for sasl configuration...
-+
-+ -+
-+/etc/postfix/master.cf:
-+    smtp-wraptls   unix   ...   ...   ...   ...   ...   smtp
-+        -o { smtp_tls_wrappermode = yes }
-+        -o { smtp_tls_security_level = encrypt }
-+
-+ -+

Example 3: Sender-dependent selection for a combination of -+MUA-to-MTA submission services. This combines examples 1 and 2 with -+examples of how to disable SRV and look up IP address records for -+"smtp-relay.example.net" and "smtp-relay.other.example". Again, -+specify the older name "smtps" instead of "submissions" when a -+provider has DNS SRV records like "_smtps._tcp.example.org" instead -+of "_submissions._tcp.example.org".

-+ -+
-+/etc/postfix/main.cf:
-+    use_srv_lookup = submission, submissions
-+    sender_dependent_default_transport_maps = inline:{
-+        # Destinations that support SRV record lookup.
-+        { user1@example.com = smtp:example.com:submission }
-+        { user2@example.org = smtp-wraptls:example.org:submissions }
-+        # Use [destination] to force IP address lookups.
-+        { user3@example.net = smtp:[smtp-relay.example.net]:submission }
-+        { user4@other.example =
-+              smtp-wraptls:[smtp-relay.other.example]:submissions } }
-+    ...see SASL_README for sasl configuration...
-+
-+ -+

Example 4: MTA-to-MTA traffic, using SRV record lookup for the -+SMTP service. This is useful for Postfix tests, and may be useful -+in environments where ports are dynamically assigned to servers. -+

-+ -+
-+/etc/postfix/main.cf:
-+    use_srv_lookup = smtp
-+    # Fall back to MX record lookup when SRV records are unavailable.
-+    #allow_srv_lookup_fallback = yes
-+    #ignore_srv_lookup_error = yes
-+
-+ -+

This feature was backported from Postfix 3.8.

-+ -+%PARAM ignore_srv_lookup_error no -+ -+

When SRV record lookup fails, fall back to MX or IP address -+lookup as if SRV record lookup was not enabled.

-+ -+

This feature was backported from Postfix 3.8.

-+ -+%PARAM allow_srv_lookup_fallback no -+ -+

When SRV record lookup fails or no SRV record exists, fall back -+to MX or IP address lookup as if SRV record lookup was not enabled.

-+ -+

This feature was backported from Postfix 3.8.

-diff --git a/src/dns/dns.h b/src/dns/dns.h -index 5f53dbc..0c1dcb3 100644 ---- a/src/dns/dns.h -+++ b/src/dns/dns.h -@@ -158,10 +158,12 @@ typedef struct DNS_RR { - unsigned short class; /* C_IN, etc. */ - unsigned int ttl; /* always */ - unsigned int dnssec_valid; /* DNSSEC validated */ -- unsigned short pref; /* T_MX only */ -+ unsigned short pref; /* T_MX and T_SRV record related */ -+ unsigned short weight; /* T_SRV related, defined in rfc2782 */ -+ unsigned short port; /* T_SRV related, defined in rfc2782 */ - struct DNS_RR *next; /* linkage */ - size_t data_len; /* actual data size */ -- char data[1]; /* actually a bunch of data */ -+ char *data; /* a bunch of data */ - } DNS_RR; - - /* -@@ -183,14 +185,29 @@ extern char *dns_strrecord(VSTRING *, DNS_RR *); - /* - * dns_rr.c - */ -+#define DNS_RR_NOPREF (0) -+#define DNS_RR_NOWEIGHT (0) -+#define DNS_RR_NOPORT (0) -+ -+#define dns_rr_create_noport(qname, rname, type, class, ttl, pref, data, \ -+ data_len) \ -+ dns_rr_create((qname), (rname), (type), (class), (ttl), \ -+ (pref), DNS_RR_NOWEIGHT, DNS_RR_NOPORT, (data), (data_len)) -+ -+#define dns_rr_create_nopref(qname, rname, type, class, ttl, data, data_len) \ -+ dns_rr_create_noport((qname), (rname), (type), (class), (ttl), \ -+ DNS_RR_NOPREF, (data), (data_len)) -+ - extern DNS_RR *dns_rr_create(const char *, const char *, - ushort, ushort, - unsigned, unsigned, -+ unsigned, unsigned, - const char *, size_t); - extern void dns_rr_free(DNS_RR *); - extern DNS_RR *dns_rr_copy(DNS_RR *); - extern DNS_RR *dns_rr_append(DNS_RR *, DNS_RR *); - extern DNS_RR *dns_rr_sort(DNS_RR *, int (*) (DNS_RR *, DNS_RR *)); -+extern DNS_RR *dns_srv_rr_sort(DNS_RR *); - extern int dns_rr_compare_pref_ipv6(DNS_RR *, DNS_RR *); - extern int dns_rr_compare_pref_ipv4(DNS_RR *, DNS_RR *); - extern int dns_rr_compare_pref_any(DNS_RR *, DNS_RR *); -@@ -295,8 +312,9 @@ extern int dns_get_h_errno(void); - * Below is the precedence order. The order between DNS_RETRY and DNS_NOTFOUND - * is arbitrary. - */ --#define DNS_RECURSE (-7) /* internal only: recursion needed */ --#define DNS_NOTFOUND (-6) /* query ok, data not found */ -+#define DNS_RECURSE (-8) /* internal only: recursion needed */ -+#define DNS_NOTFOUND (-7) /* query ok, data not found */ -+#define DNS_NULLSRV (-6) /* query ok, service unavailable */ - #define DNS_NULLMX (-5) /* query ok, service unavailable */ - #define DNS_FAIL (-4) /* query failed, don't retry */ - #define DNS_INVAL (-3) /* query ok, malformed reply */ -diff --git a/src/dns/dns_lookup.c b/src/dns/dns_lookup.c -index 615902d..18d6916 100644 ---- a/src/dns/dns_lookup.c -+++ b/src/dns/dns_lookup.c -@@ -740,6 +740,8 @@ static int dns_get_rr(DNS_RR **list, const char *orig_name, DNS_REPLY *reply, - int comp_len; - ssize_t data_len; - unsigned pref = 0; -+ unsigned weight = 0; -+ unsigned port = 0; - unsigned char *src; - unsigned char *dst; - int ch; -@@ -765,6 +767,18 @@ static int dns_get_rr(DNS_RR **list, const char *orig_name, DNS_REPLY *reply, - return (DNS_INVAL); - data_len = strlen(temp) + 1; - break; -+ case T_SRV: -+ GETSHORT(pref, pos); -+ GETSHORT(weight, pos); -+ GETSHORT(port, pos); -+ if (dn_expand(reply->buf, reply->end, pos, temp, sizeof(temp)) < 0) -+ return (DNS_RETRY); -+ if (*temp == 0) -+ return (DNS_NULLSRV); -+ if (!valid_rr_name(temp, "resource data", fixed->type, reply)) -+ return (DNS_INVAL); -+ data_len = strlen(temp) + 1; -+ break; - case T_MX: - GETSHORT(pref, pos); - if (dn_expand(reply->buf, reply->end, pos, temp, sizeof(temp)) < 0) -@@ -860,7 +874,7 @@ static int dns_get_rr(DNS_RR **list, const char *orig_name, DNS_REPLY *reply, - break; - } - *list = dns_rr_create(orig_name, rr_name, fixed->type, fixed->class, -- fixed->ttl, pref, tempbuf, data_len); -+ fixed->ttl, pref, weight, port, tempbuf, data_len); - return (DNS_OK); - } - -@@ -960,7 +974,7 @@ static int dns_get_answer(const char *orig_name, DNS_REPLY *reply, int type, - resource_found++; - rr->dnssec_valid = *maybe_secure ? reply->dnssec_ad : 0; - *rrlist = dns_rr_append(*rrlist, rr); -- } else if (status == DNS_NULLMX) { -+ } else if (status == DNS_NULLMX || status == DNS_NULLSRV) { - CORRUPT(status); /* TODO: use better name */ - } else if (not_found_status != DNS_RETRY) - not_found_status = status; -@@ -1094,6 +1108,12 @@ int dns_lookup_x(const char *name, unsigned type, unsigned flags, - name); - DNS_SET_H_ERRNO(&dns_res_state, NO_DATA); - return (status); -+ case DNS_NULLSRV: -+ if (why) -+ vstring_sprintf(why, "Domain %s does not support SRV requests", -+ name); -+ DNS_SET_H_ERRNO(&dns_res_state, NO_DATA); -+ return (status); - case DNS_OK: - if (rrlist && dns_rr_filter_maps) { - if (dns_rr_filter_execute(rrlist) < 0) { -diff --git a/src/dns/dns_rr.c b/src/dns/dns_rr.c -index b550788..15b5dee 100644 ---- a/src/dns/dns_rr.c -+++ b/src/dns/dns_rr.c -@@ -7,13 +7,15 @@ - /* #include - /* - /* DNS_RR *dns_rr_create(qname, rname, type, class, ttl, preference, --/* data, data_len) -+/* weight, port, data, data_len) - /* const char *qname; - /* const char *rname; - /* unsigned short type; - /* unsigned short class; - /* unsigned int ttl; - /* unsigned preference; -+/* unsigned weight; -+/* unsigned port; - /* const char *data; - /* size_t data_len; - /* -@@ -49,6 +51,30 @@ - /* DNS_RR *dns_rr_remove(list, record) - /* DNS_RR *list; - /* DNS_RR *record; -+/* -+/* DNS_RR *dns_srv_rr_sort(list) -+/* DNS_RR *list; -+/* AUXILIARY FUNCTIONS -+/* DNS_RR *dns_rr_create_nopref(qname, rname, type, class, ttl, -+/* data, data_len) -+/* const char *qname; -+/* const char *rname; -+/* unsigned short type; -+/* unsigned short class; -+/* unsigned int ttl; -+/* const char *data; -+/* size_t data_len; -+/* -+/* DNS_RR *dns_rr_create_noport(qname, rname, type, class, ttl, -+/* preference, data, data_len) -+/* const char *qname; -+/* const char *rname; -+/* unsigned short type; -+/* unsigned short class; -+/* unsigned int ttl; -+/* unsigned preference; -+/* const char *data; -+/* size_t data_len; - /* DESCRIPTION - /* The routines in this module maintain memory for DNS resource record - /* information, and maintain lists of DNS resource records. -@@ -56,10 +82,14 @@ - /* dns_rr_create() creates and initializes one resource record. - /* The \fIqname\fR field specifies the query name. - /* The \fIrname\fR field specifies the reply name. --/* \fIpreference\fR is used for MX records; \fIdata\fR is a null -+/* \fIpreference\fR is used for MX and SRV records; \fIweight\fR -+/* and \fIport\fR are used for SRV records; \fIdata\fR is a null - /* pointer or specifies optional resource-specific data; - /* \fIdata_len\fR is the amount of resource-specific data. - /* -+/* dns_rr_create_nopref() and dns_rr_create_noport() are convenience -+/* wrappers around dns_rr_create() that take fewer arguments. -+/* - /* dns_rr_free() releases the resource used by of zero or more - /* resource records. - /* -@@ -81,6 +111,9 @@ - /* dns_rr_remove() removes the specified record from the specified list. - /* The updated list is the result value. - /* The record MUST be a list member. -+/* -+/* dns_srv_rr_sort() sorts a list of SRV records according to -+/* their priority and weight as described in RFC 2782. - /* LICENSE - /* .ad - /* .fi -@@ -113,11 +146,15 @@ - DNS_RR *dns_rr_create(const char *qname, const char *rname, - ushort type, ushort class, - unsigned int ttl, unsigned pref, -+ unsigned weight, unsigned port, - const char *data, size_t data_len) - { - DNS_RR *rr; - -- rr = (DNS_RR *) mymalloc(sizeof(*rr) + data_len - 1); -+ /* -+ * Note: if this function is changed, update dns_rr_copy(). -+ */ -+ rr = (DNS_RR *) mymalloc(sizeof(*rr)); - rr->qname = mystrdup(qname); - rr->rname = mystrdup(rname); - rr->type = type; -@@ -125,8 +162,14 @@ DNS_RR *dns_rr_create(const char *qname, const char *rname, - rr->ttl = ttl; - rr->dnssec_valid = 0; - rr->pref = pref; -- if (data && data_len > 0) -+ rr->weight = weight; -+ rr->port = port; -+ if (data_len != 0) { -+ rr->data = mymalloc(data_len); - memcpy(rr->data, data, data_len); -+ } else { -+ rr->data = 0; -+ } - rr->data_len = data_len; - rr->next = 0; - return (rr); -@@ -141,6 +184,8 @@ void dns_rr_free(DNS_RR *rr) - dns_rr_free(rr->next); - myfree(rr->qname); - myfree(rr->rname); -+ if (rr->data) -+ myfree(rr->data); - myfree((void *) rr); - } - } -@@ -149,16 +194,17 @@ void dns_rr_free(DNS_RR *rr) - - DNS_RR *dns_rr_copy(DNS_RR *src) - { -- ssize_t len = sizeof(*src) + src->data_len - 1; - DNS_RR *dst; - - /* -- * Combine struct assignment and data copy in one block copy operation. -+ * Note: struct copy, because dns_rr_create() would not copy all fields. - */ -- dst = (DNS_RR *) mymalloc(len); -- memcpy((void *) dst, (void *) src, len); -+ dst = (DNS_RR *) mymalloc(sizeof(*dst)); -+ *dst = *src; - dst->qname = mystrdup(src->qname); - dst->rname = mystrdup(src->rname); -+ if (dst->data) -+ dst->data = mymemdup(src->data, src->data_len); - dst->next = 0; - return (dst); - } -@@ -247,6 +293,12 @@ DNS_RR *dns_rr_sort(DNS_RR *list, int (*compar) (DNS_RR *, DNS_RR *)) - int len; - int i; - -+ /* -+ * Avoid mymalloc() panic. -+ */ -+ if (list == 0) -+ return (list); -+ - /* - * Save state and initialize. - */ -@@ -293,6 +345,12 @@ DNS_RR *dns_rr_shuffle(DNS_RR *list) - int i; - int r; - -+ /* -+ * Avoid mymalloc() panic. -+ */ -+ if (list == 0) -+ return (list); -+ - /* - * Build linear array with pointers to each list element. - */ -@@ -345,3 +403,141 @@ DNS_RR *dns_rr_remove(DNS_RR *list, DNS_RR *record) - } - return (list); - } -+ -+/* weight_order - sort equal-priority records by weight */ -+ -+static void weight_order(DNS_RR **array, int count) -+{ -+ int unordered_weights; -+ int i; -+ -+ /* -+ * Compute the sum of record weights. If weights are not supplied then -+ * this function would be a noop. In fact this would be a noop when all -+ * weights have the same value, whether that weight is zero or not. There -+ * is no need to give special treatment to zero weights. -+ */ -+ for (unordered_weights = 0, i = 0; i < count; i++) -+ unordered_weights += array[i]->weight; -+ if (unordered_weights == 0) -+ return; -+ -+ /* -+ * The record ordering code below differs from RFC 2782 when the input -+ * contains a mix of zero and non-zero weights: the code below does not -+ * give special treatment to zero weights. Instead, it treats a zero -+ * weight just like any other small weight. Fewer special cases make for -+ * code that is simpler and more robust. -+ */ -+ for (i = 0; i < count - 1; i++) { -+ int running_sum; -+ int threshold; -+ int k; -+ DNS_RR *temp; -+ -+ /* -+ * Choose a random threshold [0..unordered_weights] inclusive. -+ */ -+ threshold = myrand() % (unordered_weights + 1); -+ -+ /* -+ * Move the first record with running_sum >= threshold to the ordered -+ * list, and update unordered_weights. -+ */ -+ for (running_sum = 0, k = i; k < count; k++) { -+ running_sum += array[k]->weight; -+ if (running_sum >= threshold) { -+ unordered_weights -= array[k]->weight; -+ temp = array[i]; -+ array[i] = array[k]; -+ array[k] = temp; -+ break; -+ } -+ } -+ } -+} -+ -+/* dns_srv_rr_sort - sort resource record list */ -+ -+DNS_RR *dns_srv_rr_sort(DNS_RR *list) -+{ -+ int (*saved_user) (DNS_RR *, DNS_RR *); -+ DNS_RR **rr_array; -+ DNS_RR *rr; -+ int len; -+ int i; -+ int r; -+ int cur_pref; -+ int left_bound; /* inclusive */ -+ int right_bound; /* non-inclusive */ -+ -+ /* -+ * Avoid mymalloc() panic, or rr_array[0] fence-post error. -+ */ -+ if (list == 0) -+ return (list); -+ -+ /* -+ * Save state and initialize. -+ */ -+ saved_user = dns_rr_sort_user; -+ dns_rr_sort_user = dns_rr_compare_pref_any; -+ -+ /* -+ * Build linear array with pointers to each list element. -+ */ -+ for (len = 0, rr = list; rr != 0; len++, rr = rr->next) -+ /* void */ ; -+ rr_array = (DNS_RR **) mymalloc(len * sizeof(*rr_array)); -+ for (len = 0, rr = list; rr != 0; len++, rr = rr->next) -+ rr_array[len] = rr; -+ -+ /* -+ * Shuffle resource records. Every element has an equal chance of landing -+ * in slot 0. After that every remaining element has an equal chance of -+ * landing in slot 1, ... This is exactly n! states for n! permutations. -+ */ -+ for (i = 0; i < len - 1; i++) { -+ r = i + (myrand() % (len - i)); /* Victor&Son */ -+ rr = rr_array[i]; -+ rr_array[i] = rr_array[r]; -+ rr_array[r] = rr; -+ } -+ -+ /* First order the records by preference. */ -+ qsort((void *) rr_array, len, sizeof(*rr_array), dns_rr_sort_callback); -+ -+ /* -+ * Walk through records and sort the records in every same-preference -+ * partition according to their weight. Note that left_bound is -+ * inclusive, and that right-bound is non-inclusive. -+ */ -+ left_bound = 0; -+ cur_pref = rr_array[left_bound]->pref; /* assumes len > 0 */ -+ -+ for (right_bound = 1; /* see below */ ; right_bound++) { -+ if (right_bound == len || rr_array[right_bound]->pref != cur_pref) { -+ if (right_bound - left_bound > 1) -+ weight_order(rr_array + left_bound, right_bound - left_bound); -+ if (right_bound == len) -+ break; -+ left_bound = right_bound; -+ cur_pref = rr_array[left_bound]->pref; -+ } -+ } -+ -+ /* -+ * Fix the links. -+ */ -+ for (i = 0; i < len - 1; i++) -+ rr_array[i]->next = rr_array[i + 1]; -+ rr_array[i]->next = 0; -+ list = rr_array[0]; -+ -+ /* -+ * Cleanup. -+ */ -+ myfree((void *) rr_array); -+ dns_rr_sort_user = saved_user; -+ return (list); -+} -diff --git a/src/dns/dns_sa_to_rr.c b/src/dns/dns_sa_to_rr.c -index 6b9efcc..b5dee20 100644 ---- a/src/dns/dns_sa_to_rr.c -+++ b/src/dns/dns_sa_to_rr.c -@@ -55,14 +55,14 @@ DNS_RR *dns_sa_to_rr(const char *hostname, unsigned pref, struct sockaddr *sa) - #define DUMMY_TTL 0 - - if (sa->sa_family == AF_INET) { -- return (dns_rr_create(hostname, hostname, T_A, C_IN, DUMMY_TTL, pref, -- (char *) &SOCK_ADDR_IN_ADDR(sa), -- sizeof(SOCK_ADDR_IN_ADDR(sa)))); -+ return (dns_rr_create_noport(hostname, hostname, T_A, C_IN, DUMMY_TTL, -+ pref, (char *) &SOCK_ADDR_IN_ADDR(sa), -+ sizeof(SOCK_ADDR_IN_ADDR(sa)))); - #ifdef HAS_IPV6 - } else if (sa->sa_family == AF_INET6) { -- return (dns_rr_create(hostname, hostname, T_AAAA, C_IN, DUMMY_TTL, pref, -- (char *) &SOCK_ADDR_IN6_ADDR(sa), -- sizeof(SOCK_ADDR_IN6_ADDR(sa)))); -+ return (dns_rr_create_noport(hostname, hostname, T_AAAA, C_IN, DUMMY_TTL, -+ pref, (char *) &SOCK_ADDR_IN6_ADDR(sa), -+ sizeof(SOCK_ADDR_IN6_ADDR(sa)))); - #endif - } else { - errno = EAFNOSUPPORT; -@@ -121,7 +121,7 @@ int main(int argc, char **argv) - resv[len++] = res; - qsort((void *) resv, len, sizeof(*resv), compare_family); - for (n = 0; n < len; n++) { -- if ((rr = dns_sa_to_rr(argv[0], 0, resv[n]->ai_addr)) == 0) -+ if ((rr = dns_sa_to_rr(argv[0], DNS_RR_NOPREF, resv[n]->ai_addr)) == 0) - msg_fatal("dns_sa_to_rr: %m"); - if (dns_rr_to_pa(rr, &hostaddr) == 0) - msg_fatal("dns_rr_to_pa: %m"); -diff --git a/src/dns/dns_strrecord.c b/src/dns/dns_strrecord.c -index 6b8e989..1e3b743 100644 ---- a/src/dns/dns_strrecord.c -+++ b/src/dns/dns_strrecord.c -@@ -80,6 +80,10 @@ char *dns_strrecord(VSTRING *buf, DNS_RR *rr) - case T_MX: - vstring_sprintf_append(buf, "%u %s.", rr->pref, rr->data); - break; -+ case T_SRV: -+ vstring_sprintf_append(buf, "%u %u %u %s.", rr->pref, rr->weight, -+ rr->port, rr->data); -+ break; - case T_TLSA: - if (rr->data_len >= 3) { - uint8_t *ip = (uint8_t *) rr->data; -diff --git a/src/dns/dns_strtype.c b/src/dns/dns_strtype.c -index 70e59ac..7eebe3c 100644 ---- a/src/dns/dns_strtype.c -+++ b/src/dns/dns_strtype.c -@@ -180,6 +180,9 @@ static struct dns_type_map dns_type_map[] = { - #ifdef T_ANY - T_ANY, "ANY", - #endif -+#ifdef T_SRV -+ T_SRV, "SRV", -+#endif - }; - - /* dns_strtype - translate DNS query type to string */ -diff --git a/src/global/mail_params.h b/src/global/mail_params.h -index 25df5fc..781264b 100644 ---- a/src/global/mail_params.h -+++ b/src/global/mail_params.h -@@ -4370,6 +4370,21 @@ extern char *var_dnssec_probe; - "lmtp=24, smtp=25, smtps=submissions=465, submission=587" - extern char *var_known_tcp_ports; - -+ /* -+ * SRV lookup support. -+ */ -+#define VAR_USE_SRV_LOOKUP "use_srv_lookup" -+#define DEF_USE_SRV_LOOKUP "" -+extern char *var_use_srv_lookup; -+ -+#define VAR_IGN_SRV_LOOKUP_ERR "ignore_srv_lookup_error" -+#define DEF_IGN_SRV_LOOKUP_ERR 0 -+extern bool var_ign_srv_lookup_err; -+ -+#define VAR_ALLOW_SRV_FALLBACK "allow_srv_lookup_fallback" -+#define DEF_ALLOW_SRV_FALLBACK 0 -+extern bool var_allow_srv_fallback; -+ - /* LICENSE - /* .ad - /* .fi -diff --git a/src/posttls-finger/posttls-finger.c b/src/posttls-finger/posttls-finger.c -index 502645c..7a62afe 100644 ---- a/src/posttls-finger/posttls-finger.c -+++ b/src/posttls-finger/posttls-finger.c -@@ -237,6 +237,8 @@ - /* is encountered, up to 5 times or as specified with the \fB-m\fR option. - /* By default reconnection is disabled, specify a positive delay to - /* enable this behavior. -+/* .IP "\fB-R\fR" -+/* Use SRV lookup instead of MX. - /* .IP "\fB-s \fIservername\fR" - /* The server name to send with the TLS Server Name Indication (SNI) - /* extension. When the server has DANE TLSA records, this parameter -@@ -468,6 +470,7 @@ typedef struct STATE { - DNS_RR *mx; /* MX RRset qname, rname, valid */ - int pass; /* Pass number, 2 for reconnect */ - int nochat; /* disable chat logging */ -+ int dosrv; /* look up SRV records instead of MX */ - char *helo; /* Server name from EHLO reply */ - DSN_BUF *why; /* SMTP-style error message */ - VSTRING *buffer; /* Response buffer */ -@@ -1158,7 +1161,7 @@ static VSTREAM *connect_addr(STATE *state, DNS_RR *addr) - /* addr_one - address lookup for one host name */ - - static DNS_RR *addr_one(STATE *state, DNS_RR *addr_list, const char *host, -- int res_opt, unsigned pref) -+ int res_opt, unsigned pref, unsigned port) - { - static const char *myname = "addr_one"; - DSN_BUF *why = state->why; -@@ -1181,6 +1184,8 @@ static DNS_RR *addr_one(STATE *state, DNS_RR *addr_list, const char *host, - if ((addr = dns_sa_to_rr(host, pref, res0->ai_addr)) == 0) - msg_fatal("host %s: conversion error for address family %d: %m", - host, ((struct sockaddr *) (res0->ai_addr))->sa_family); -+ addr->pref = pref; -+ addr->port = port; - addr_list = dns_rr_append(addr_list, addr); - freeaddrinfo(res0); - return (addr_list); -@@ -1197,8 +1202,10 @@ static DNS_RR *addr_one(STATE *state, DNS_RR *addr_list, const char *host, - why->reason, DNS_REQ_FLAG_NONE, - proto_info->dns_atype_list)) { - case DNS_OK: -- for (rr = addr; rr; rr = rr->next) -+ for (rr = addr; rr; rr = rr->next) { - rr->pref = pref; -+ rr->port = port; -+ } - addr_list = dns_rr_append(addr_list, addr); - return (addr_list); - default: -@@ -1285,15 +1292,15 @@ static DNS_RR *mx_addr_list(STATE *state, DNS_RR *mx_names) - #endif - - for (rr = mx_names; rr; rr = rr->next) { -- if (rr->type != T_MX) -+ if (rr->type != T_MX && rr->type != T_SRV) - msg_panic("%s: bad resource type: %d", myname, rr->type); - addr_list = addr_one(state, addr_list, (char *) rr->data, res_opt, -- rr->pref); -+ rr->pref, rr->port); - } - return (addr_list); - } - --/* smtp_domain_addr - mail exchanger address lookup */ -+/* domain_addr - mail exchanger address lookup */ - - static DNS_RR *domain_addr(STATE *state, char *domain) - { -@@ -1358,6 +1365,74 @@ static DNS_RR *domain_addr(STATE *state, char *domain) - return (addr_list); - } - -+/* service_addr - mail exchanger address lookup */ -+ -+static DNS_RR *service_addr(STATE *state, const char *domain, -+ const char *service) -+{ -+ VSTRING *srv_qname = vstring_alloc(100); -+ char *str_srv_qname; -+ DNS_RR *srv_names; -+ DNS_RR *addr_list = 0; -+ int r = 0; /* Resolver flags */ -+ const char *aname; -+ -+ dsb_reset(state->why); -+ -+#if (RES_USE_DNSSEC != 0) && (RES_USE_EDNS0 != 0) -+ r |= RES_USE_DNSSEC; -+#endif -+ -+ vstring_sprintf(srv_qname, "_%s._tcp.%s", service, domain); -+ str_srv_qname = STR(srv_qname); -+ -+ /* -+ * IDNA support. -+ */ -+#ifndef NO_EAI -+ if (!allascii(str_srv_qname) -+ && (aname = midna_domain_to_ascii(str_srv_qname)) != 0) { -+ msg_info("%s asciified to %s", str_srv_qname, aname); -+ } else -+#endif -+ aname = str_srv_qname; -+ -+ switch (dns_lookup(aname, T_SRV, r, &srv_names, (VSTRING *) 0, -+ state->why->reason)) { -+ default: -+ dsb_status(state->why, "4.4.3"); -+ break; -+ case DNS_INVAL: -+ dsb_status(state->why, "5.4.4"); -+ break; -+ case DNS_NULLMX: -+ dsb_status(state->why, "5.1.0"); -+ break; -+ case DNS_FAIL: -+ dsb_status(state->why, "5.4.3"); -+ break; -+ case DNS_OK: -+ /* Shuffle then sort the SRV rr records by priority and weight. */ -+ srv_names = dns_srv_rr_sort(srv_names); -+ addr_list = mx_addr_list(state, srv_names); -+ state->mx = dns_rr_copy(srv_names); -+ dns_rr_free(srv_names); -+ if (addr_list == 0) { -+ msg_warn("no SRV host for %s has a valid address record", -+ str_srv_qname); -+ break; -+ } -+ /* TODO: sort by priority, weight, and address family preference. */ -+ break; -+ case DNS_NOTFOUND: -+ dsb_status(state->why, "5.4.4"); -+ break; -+ } -+ -+ vstring_free(srv_qname); -+ return (addr_list); -+} -+ - /* host_addr - direct host lookup */ - - static DNS_RR *host_addr(STATE *state, const char *host) -@@ -1384,7 +1459,8 @@ static DNS_RR *host_addr(STATE *state, const char *host) - ahost = host; - - #define PREF0 0 -- addr_list = addr_one(state, (DNS_RR *) 0, ahost, res_opt, PREF0); -+#define NOPORT 0 -+ addr_list = addr_one(state, (DNS_RR *) 0, ahost, res_opt, PREF0, NOPORT); - if (addr_list && addr_list->next) { - addr_list = dns_rr_shuffle(addr_list); - if (inet_proto_info()->ai_family_list[1] != 0) -@@ -1468,7 +1544,8 @@ static int dane_host_level(STATE *state, DNS_RR *addr) - /* parse_destination - parse host/port destination */ - - static char *parse_destination(char *destination, char *def_service, -- char **hostp, unsigned *portp) -+ char **hostp, char **servicep, -+ unsigned *portp) - { - char *buf = mystrdup(destination); - char *service; -@@ -1484,13 +1561,13 @@ static char *parse_destination(char *destination, char *def_service, - * Parse the host/port information. We're working with a copy of the - * destination argument so the parsing can be destructive. - */ -- if ((err = host_port(buf, hostp, (char *) 0, &service, def_service)) != 0) -+ if ((err = host_port(buf, hostp, (char *) 0, servicep, def_service)) != 0) - msg_fatal("%s in server description: %s", err, destination); - - /* - * Convert service to port number, network byte order. - */ -- service = (char *) filter_known_tcp_port(service); -+ service = (char *) filter_known_tcp_port(*servicep); - if (alldig(service)) { - if ((port = atoi(service)) >= 65536 || port == 0) - msg_fatal("bad network port: %s for destination: %s", -@@ -1512,17 +1589,21 @@ static char *parse_destination(char *destination, char *def_service, - static void connect_remote(STATE *state, char *dest) - { - DNS_RR *addr; -- char *buf; -- char *domain; - - /* When reconnecting use IP address of previous session */ - if (state->addr == 0) { -+ char *buf; -+ char *domain; -+ char *service; -+ - buf = parse_destination(dest, state->smtp ? "smtp" : "24", -- &domain, &state->port); -+ &domain, &service, &state->port); - if (!state->nexthop) - state->nexthop = mystrdup(domain); - if (state->smtp == 0 || *dest == '[') - state->addr = host_addr(state, domain); -+ else if (state->dosrv) -+ state->addr = service_addr(state, domain, service); - else - state->addr = domain_addr(state, domain); - myfree(buf); -@@ -1536,10 +1617,14 @@ static void connect_remote(STATE *state, char *dest) - for (addr = state->addr; addr; addr = addr->next) { - int level = dane_host_level(state, addr); - -+ if (addr->port) /* SRV port override */ -+ state->port = htons(addr->port); -+ - if (level == TLS_LEV_INVALID - || (state->stream = connect_addr(state, addr)) == 0) { -- msg_info("Failed to establish session to %s via %s: %s", -- dest, HNAME(addr), vstring_str(state->why->reason)); -+ msg_info("Failed to establish session to %s via %s:%u: %s", -+ dest, HNAME(addr), addr->port, -+ vstring_str(state->why->reason)); - continue; - } - /* We have a connection */ -@@ -1807,6 +1892,7 @@ static void parse_options(STATE *state, int argc, char *argv[]) - - state->smtp = 1; - state->pass = 1; -+ state->dosrv = 0; - state->reconnect = -1; - state->max_reconnect = 5; - state->wrapper_mode = 0; -@@ -1817,7 +1903,7 @@ static void parse_options(STATE *state, int argc, char *argv[]) - memset((void *) &state->options, 0, sizeof(state->options)); - state->options.host_lookup = mystrdup("dns"); - --#define OPTS "a:ch:o:St:T:v" -+#define OPTS "a:ch:o:RSt:T:v" - #ifdef USE_TLS - #define TLSOPTS "A:Cd:fF:g:H:k:K:l:L:m:M:p:P:r:s:wX" - -@@ -1856,6 +1942,9 @@ static void parse_options(STATE *state, int argc, char *argv[]) - case 'o': - override(optarg); - break; -+ case 'R': -+ state->dosrv = 1; -+ break; - case 'S': - state->smtp = 0; - break; -diff --git a/src/smtp/lmtp_params.c b/src/smtp/lmtp_params.c -index cc33646..bca7cd4 100644 ---- a/src/smtp/lmtp_params.c -+++ b/src/smtp/lmtp_params.c -@@ -65,6 +65,7 @@ - VAR_LMTP_DNS_RE_FILTER, DEF_LMTP_DNS_RE_FILTER, &var_smtp_dns_re_filter, 0, 0, - VAR_TLSPROXY_SERVICE, DEF_TLSPROXY_SERVICE, &var_tlsproxy_service, 1, 0, - VAR_HFROM_FORMAT, DEF_HFROM_FORMAT, &var_hfrom_format, 1, 0, -+ VAR_USE_SRV_LOOKUP, DEF_USE_SRV_LOOKUP, &var_use_srv_lookup, 0, 0, - 0, - }; - static const CONFIG_TIME_TABLE lmtp_time_table[] = { -@@ -128,6 +129,8 @@ - VAR_LMTP_DUMMY_MAIL_AUTH, DEF_LMTP_DUMMY_MAIL_AUTH, &var_smtp_dummy_mail_auth, - VAR_LMTP_BALANCE_INET_PROTO, DEF_LMTP_BALANCE_INET_PROTO, &var_smtp_balance_inet_proto, - VAR_LMTP_BIND_ADDR_ENFORCE, DEF_LMTP_BIND_ADDR_ENFORCE, &var_smtp_bind_addr_enforce, -+ VAR_IGN_SRV_LOOKUP_ERR, DEF_IGN_SRV_LOOKUP_ERR, &var_ign_srv_lookup_err, -+ VAR_ALLOW_SRV_FALLBACK, DEF_ALLOW_SRV_FALLBACK, &var_allow_srv_fallback, - 0, - }; - static const CONFIG_NBOOL_TABLE lmtp_nbool_table[] = { -diff --git a/src/smtp/smtp.c b/src/smtp/smtp.c -index 791ec89..816095b 100644 ---- a/src/smtp/smtp.c -+++ b/src/smtp/smtp.c -@@ -146,6 +146,7 @@ - /* RFC 2046 (MIME: Media Types) - /* RFC 2554 (AUTH command) - /* RFC 2821 (SMTP protocol) -+/* RFC 2782 (SRV resource records) - /* RFC 2920 (SMTP Pipelining) - /* RFC 3207 (STARTTLS command) - /* RFC 3461 (SMTP DSN Extension) -@@ -352,6 +353,17 @@ - /* DATA requests, when deadlines are enabled with smtp_per_request_deadline. - /* .IP "\fBheader_from_format (standard)\fR" - /* The format of the Postfix-generated \fBFrom:\fR header. -+/* .PP -+/* Backported from Postfix version 3.8: -+/* .IP "\fBuse_srv_lookup (empty)\fR" -+/* Enables discovery for the specified service(s) using DNS SRV -+/* records. -+/* .IP "\fBignore_srv_lookup_error (no)\fR" -+/* When SRV record lookup fails, fall back to MX or IP address -+/* lookup as if SRV record lookup was not enabled. -+/* .IP "\fBallow_srv_lookup_fallback (no)\fR" -+/* When SRV record lookup fails or no SRV record exists, fall back -+/* to MX or IP address lookup as if SRV record lookup was not enabled. - /* MIME PROCESSING CONTROLS - /* .ad - /* .fi -@@ -1092,6 +1104,9 @@ char *var_smtp_dns_re_filter; - bool var_smtp_balance_inet_proto; - bool var_smtp_req_deadline; - int var_smtp_min_data_rate; -+char *var_use_srv_lookup; -+bool var_ign_srv_lookup_err; -+bool var_allow_srv_fallback; - - /* Special handling of 535 AUTH errors. */ - char *var_smtp_sasl_auth_cache_name; -@@ -1118,6 +1133,7 @@ HBC_CHECKS *smtp_header_checks; /* limited header checks */ - HBC_CHECKS *smtp_body_checks; /* limited body checks */ - SMTP_CLI_ATTR smtp_cli_attr; /* parsed command-line */ - int smtp_hfrom_format; /* postmaster notifications */ -+STRING_LIST *smtp_use_srv_lookup; - - #ifdef USE_TLS - -@@ -1408,6 +1424,15 @@ static void post_init(char *unused_name, char **argv) - * header_from format, for postmaster notifications. - */ - smtp_hfrom_format = hfrom_format_parse(VAR_HFROM_FORMAT, var_hfrom_format); -+ -+ -+ /* -+ * Service discovery with SRV record lookup. -+ */ -+ if (*var_use_srv_lookup) -+ smtp_use_srv_lookup = string_list_init(VAR_USE_SRV_LOOKUP, -+ MATCH_FLAG_RETURN, -+ var_use_srv_lookup); - } - - /* pre_init - pre-jail initialization */ -diff --git a/src/smtp/smtp.h b/src/smtp/smtp.h -index 0d5c80a..2818d6e 100644 ---- a/src/smtp/smtp.h -+++ b/src/smtp/smtp.h -@@ -84,6 +84,14 @@ typedef struct SMTP_ITERATOR { - vstring_strcpy((iter)->dest, STR((iter)->saved_dest)); \ - } while (0) - -+#define SMTP_ITER_UPDATE_HOST(iter, _host, _addr, _rr) do { \ -+ vstring_strcpy((iter)->host, (_host)); \ -+ vstring_strcpy((iter)->addr, (_addr)); \ -+ (iter)->rr = (_rr); \ -+ if ((_rr)->port) \ -+ (iter)->port = htons((_rr)->port); /* SRV port override */ \ -+ } while (0) -+ - /* - * TLS Policy support. - */ -@@ -273,6 +281,7 @@ typedef struct SMTP_STATE { - #define SMTP_MISC_FLAG_COMPLETE_SESSION (1<<7) - #define SMTP_MISC_FLAG_PREF_IPV6 (1<<8) - #define SMTP_MISC_FLAG_PREF_IPV4 (1<<9) -+#define SMTP_MISC_FLAG_FALLBACK_SRV_TO_MX (1<<10) - - #define SMTP_MISC_FLAG_CONN_CACHE_MASK \ - (SMTP_MISC_FLAG_CONN_LOAD | SMTP_MISC_FLAG_CONN_STORE) -@@ -312,6 +321,8 @@ extern MAPS *smtp_generic_maps; /* make internal address valid */ - extern int smtp_ext_prop_mask; /* address extension propagation */ - extern unsigned smtp_dns_res_opt; /* DNS query flags */ - -+extern STRING_LIST *smtp_use_srv_lookup;/* services with SRV record lookup */ -+ - #ifdef USE_TLS - - extern TLS_APPL_STATE *smtp_tls_ctx; /* client-side TLS engine */ -diff --git a/src/smtp/smtp_addr.c b/src/smtp/smtp_addr.c -index 2b5c126..94adaa1 100644 ---- a/src/smtp/smtp_addr.c -+++ b/src/smtp/smtp_addr.c -@@ -17,6 +17,15 @@ - /* char *name; - /* int misc_flags; - /* DSN_BUF *why; -+/* -+/* DNS_RR *smtp_service_addr(name, service, mxrr, misc_flags, why, -+/* found_myself) -+/* const char *name; -+/* const char *service; -+/* DNS_RR **mxrr; -+/* int misc_flags; -+/* DSN_BUF *why; -+/* int *found_myself; - /* DESCRIPTION - /* This module implements Internet address lookups. By default, - /* lookups are done via the Internet domain name service (DNS). -@@ -33,6 +42,8 @@ - /* destination. If MX records were found, the rname, qname, - /* and dnssec validation status of the MX RRset are returned - /* via mxrr, which the caller must free with dns_rr_free(). -+/* Fallback from MX to address lookups is governed by RFC 2821, -+/* and by local policy (var_ign_mx_lookup_err). - /* - /* When no mail exchanger is listed in the DNS for \fIname\fR, the - /* request is passed to smtp_host_addr(). -@@ -44,8 +55,18 @@ - /* host. The host can be specified as a numerical Internet network - /* address, or as a symbolic host name. - /* --/* Results from smtp_domain_addr() or smtp_host_addr() are --/* destroyed by dns_rr_free(), including null lists. -+/* smtp_service_addr() looks up addresses for hosts specified -+/* in SRV records for the specified domain and service. This -+/* supports the features of smtp_domain_addr() except that -+/* the order of SRV records is determined by RFC 2782, and -+/* that address records are not sorted by IP address family -+/* preference. Fallback from SRV to MX or address lookups is -+/* governed by local policy (var_ign_mx_lookup_err and -+/* var_allow_srv_fallback). -+/* -+/* Results from smtp_domain_addr(), smtp_host_addr(), and -+/* smtp_service_addr() are destroyed by dns_rr_free(), including -+/* null lists. - /* DIAGNOSTICS - /* Panics: interface violations. For example, calling smtp_domain_addr() - /* when DNS lookups are explicitly disabled. -@@ -130,7 +151,8 @@ static void smtp_print_addr(const char *what, DNS_RR *addr_list) - /* smtp_addr_one - address lookup for one host name */ - - static DNS_RR *smtp_addr_one(DNS_RR *addr_list, const char *host, int res_opt, -- unsigned pref, DSN_BUF *why) -+ unsigned pref, unsigned port, -+ DSN_BUF *why) - { - const char *myname = "smtp_addr_one"; - DNS_RR *addr = 0; -@@ -153,6 +175,8 @@ static DNS_RR *smtp_addr_one(DNS_RR *addr_list, const char *host, int res_opt, - if ((addr = dns_sa_to_rr(host, pref, res0->ai_addr)) == 0) - msg_fatal("host %s: conversion error for address family " - "%d: %m", host, res0->ai_addr->sa_family); -+ addr->pref = pref; -+ addr->port = port; - addr_list = dns_rr_append(addr_list, addr); - if (msg_verbose) - msg_info("%s: using numerical host %s", myname, host); -@@ -174,8 +198,10 @@ static DNS_RR *smtp_addr_one(DNS_RR *addr_list, const char *host, int res_opt, - why->reason, DNS_REQ_FLAG_NONE, - proto_info->dns_atype_list)) { - case DNS_OK: -- for (rr = addr; rr; rr = rr->next) -+ for (rr = addr; rr; rr = rr->next) { - rr->pref = pref; -+ rr->port = port; -+ } - addr_list = dns_rr_append(addr_list, addr); - return (addr_list); - default: -@@ -293,10 +319,10 @@ static DNS_RR *smtp_addr_list(DNS_RR *mx_names, DSN_BUF *why) - * tweaking the in-process resolver flags. - */ - for (rr = mx_names; rr; rr = rr->next) { -- if (rr->type != T_MX) -+ if (rr->type != T_MX && rr->type != T_SRV) - msg_panic("smtp_addr_list: bad resource type: %d", rr->type); - addr_list = smtp_addr_one(addr_list, (char *) rr->data, res_opt, -- rr->pref, why); -+ rr->pref, rr->port, why); - } - return (addr_list); - } -@@ -678,7 +704,7 @@ DNS_RR *smtp_host_addr(const char *host, int misc_flags, DSN_BUF *why) - * address to internal form. Otherwise, the host is specified by name. - */ - #define PREF0 0 -- addr_list = smtp_addr_one((DNS_RR *) 0, ahost, res_opt, PREF0, why); -+ addr_list = smtp_addr_one((DNS_RR *) 0, ahost, res_opt, PREF0, 0, why); - if (addr_list - && (misc_flags & SMTP_MISC_FLAG_LOOP_DETECT) - && smtp_find_self(addr_list) != 0) { -@@ -700,3 +726,135 @@ DNS_RR *smtp_host_addr(const char *host, int misc_flags, DSN_BUF *why) - smtp_print_addr(host, addr_list); - return (addr_list); - } -+ -+/* smtp_service_addr - service address lookup */ -+ -+DNS_RR *smtp_service_addr(const char *name, const char *service, DNS_RR **mxrr, -+ int misc_flags, DSN_BUF *why, -+ int *found_myself) -+{ -+ static VSTRING *srv_qname = 0; -+ const char *str_srv_qname; -+ DNS_RR *srv_names = 0; -+ DNS_RR *addr_list = 0; -+ DNS_RR *self = 0; -+ unsigned best_pref; -+ unsigned best_found; -+ int r = 0; -+ const char *aname; -+ int allow_non_srv_fallback = var_allow_srv_fallback; -+ -+ dsb_reset(why); -+ -+ /* -+ * Sanity check. -+ */ -+ if (smtp_dns_support == SMTP_DNS_DISABLED) -+ msg_panic("smtp_service_addr: DNS lookup is disabled"); -+ -+ if (smtp_dns_support == SMTP_DNS_DNSSEC) { -+ r |= RES_USE_DNSSEC; -+ } -+ if (srv_qname == 0) -+ srv_qname = vstring_alloc(100); -+ vstring_sprintf(srv_qname, "_%s._tcp.%s", service, name); -+ str_srv_qname = STR(srv_qname); -+ -+ /* -+ * IDNA support. -+ */ -+#ifndef NO_EAI -+ if (!allascii(str_srv_qname) -+ && (aname = midna_domain_to_ascii(str_srv_qname)) != 0) { -+ if (msg_verbose) -+ msg_info("%s asciified to %s", str_srv_qname, aname); -+ } else -+#endif -+ aname = str_srv_qname; -+ -+ switch (dns_lookup(aname, T_SRV, r, &srv_names, (VSTRING *) 0, -+ why->reason)) { -+ default: -+ dsb_status(why, "4.4.3"); -+ allow_non_srv_fallback |= var_ign_srv_lookup_err; -+ break; -+ case DNS_INVAL: -+ dsb_status(why, "5.4.4"); -+ allow_non_srv_fallback |= var_ign_srv_lookup_err; -+ break; -+ case DNS_POLICY: -+ dsb_status(why, "4.7.0"); -+ break; -+ case DNS_FAIL: -+ dsb_status(why, "5.4.3"); -+ allow_non_srv_fallback |= var_ign_srv_lookup_err; -+ break; -+ case DNS_NULLSRV: -+ dsb_status(why, "5.1.0"); -+ break; -+ case DNS_OK: -+ /* Shuffle then sort the SRV rr records by priority and weight. */ -+ srv_names = dns_srv_rr_sort(srv_names); -+ best_pref = (srv_names ? srv_names->pref : IMPOSSIBLE_PREFERENCE); -+ addr_list = smtp_addr_list(srv_names, why); -+ if (mxrr) -+ *mxrr = dns_rr_copy(srv_names); /* copies one record! */ -+ dns_rr_free(srv_names); -+ if (addr_list == 0) { -+ msg_warn("no SRV host for %s has a valid address record", -+ str_srv_qname); -+ break; -+ } -+ /* Optional loop prevention, similar to smtp_domain_addr(). */ -+ best_found = (addr_list ? addr_list->pref : IMPOSSIBLE_PREFERENCE); -+ if (msg_verbose) -+ smtp_print_addr(aname, addr_list); -+ if ((misc_flags & SMTP_MISC_FLAG_LOOP_DETECT) -+ && (self = smtp_find_self(addr_list)) != 0) { -+ addr_list = smtp_truncate_self(addr_list, self->pref); -+ if (addr_list == 0) { -+ if (best_pref != best_found) { -+ dsb_simple(why, "4.4.4", -+ "unable to find primary relay for %s", -+ str_srv_qname); -+ } else { -+ dsb_simple(why, "5.4.6", "mail for %s loops back to myself", -+ str_srv_qname); -+ } -+ } -+ } -+ /* TODO: sort by priority, weight, and address family preference. */ -+ -+ /* Optional address family balancing, as in smtp_domain_addr(). */ -+ if (addr_list && addr_list->next) { -+ if (var_smtp_mxaddr_limit > 0 && var_smtp_balance_inet_proto) -+ addr_list = smtp_balance_inet_proto(addr_list, misc_flags, -+ var_smtp_mxaddr_limit); -+ } -+ break; -+ case DNS_NOTFOUND: -+ dsb_status(why, "5.4.4"); -+ break; -+ } -+ -+ /* -+ * If permitted, fall back to non-SRV record lookups. -+ */ -+ if (addr_list == 0 && allow_non_srv_fallback) { -+ msg_info("skipping SRV lookup for %s: %s", -+ str_srv_qname, STR(why->reason)); -+ if (misc_flags & SMTP_MISC_FLAG_FALLBACK_SRV_TO_MX) -+ addr_list = smtp_domain_addr(name, mxrr, misc_flags, why, -+ found_myself); -+ else -+ addr_list = smtp_host_addr(name, misc_flags, why); -+ } -+ -+ /* -+ * Only if we're not falling back. -+ */ -+ else { -+ *found_myself |= (self != 0); -+ } -+ return (addr_list); -+} -diff --git a/src/smtp/smtp_addr.h b/src/smtp/smtp_addr.h -index 8f20961..3d70413 100644 ---- a/src/smtp/smtp_addr.h -+++ b/src/smtp/smtp_addr.h -@@ -18,6 +18,7 @@ - */ - extern DNS_RR *smtp_host_addr(const char *, int, DSN_BUF *); - extern DNS_RR *smtp_domain_addr(const char *, DNS_RR **, int, DSN_BUF *, int *); -+extern DNS_RR *smtp_service_addr(const char *, const char *, DNS_RR **, int, DSN_BUF *, int *); - - /* LICENSE - /* .ad -diff --git a/src/smtp/smtp_connect.c b/src/smtp/smtp_connect.c -index ed58180..68faca1 100644 ---- a/src/smtp/smtp_connect.c -+++ b/src/smtp/smtp_connect.c -@@ -28,7 +28,8 @@ - /* destinations may be specified as "unix:pathname", "inet:host" - /* or "inet:host:port". - /* --/* With SMTP, the Internet domain name service is queried for mail -+/* With SMTP, or with SRV record lookup enabled, the Internet -+/* domain name service is queried for mail - /* exchanger hosts. Quote the domain name with `[' and `]' to - /* suppress mail exchanger lookups. - /* -@@ -357,7 +358,8 @@ static SMTP_SESSION *smtp_connect_sock(int sock, struct sockaddr *sa, - /* smtp_parse_destination - parse host/port destination */ - - static char *smtp_parse_destination(char *destination, char *def_service, -- char **hostp, unsigned *portp) -+ char **hostp, char **servicep, -+ unsigned *portp) - { - char *buf = mystrdup(destination); - char *service; -@@ -373,13 +375,13 @@ static char *smtp_parse_destination(char *destination, char *def_service, - * Parse the host/port information. We're working with a copy of the - * destination argument so the parsing can be destructive. - */ -- if ((err = host_port(buf, hostp, (char *) 0, &service, def_service)) != 0) -+ if ((err = host_port(buf, hostp, (char *) 0, servicep, def_service)) != 0) - msg_fatal("%s in server description: %s", err, destination); - - /* - * Convert service to port number, network byte order. - */ -- service = (char *) filter_known_tcp_port(service); -+ service = (char *) filter_known_tcp_port(*servicep); - if (alldig(service)) { - if ((port = atoi(service)) >= 65536 || port == 0) - msg_fatal("bad network port: %s for destination: %s", -@@ -661,6 +663,9 @@ static void smtp_update_addr_list(DNS_RR **addr_list, const char *server_addr, - * XXX Extend the SMTP_SESSION structure with sockaddr information so that - * we can avoid repeated string->binary transformations for the same - * address. -+ * -+ * XXX SRV support: this should match the port, too, otherwise we may -+ * eliminate too many list entries. - */ - if ((aierr = hostaddr_to_sockaddr(server_addr, (char *) 0, 0, &res0)) != 0) { - msg_warn("hostaddr_to_sockaddr %s: %s", -@@ -691,6 +696,18 @@ static int smtp_reuse_session(SMTP_STATE *state, DNS_RR **addr_list, - DSN_BUF *why = state->why; - - /* -+ * This code is called after server address/port lookup, before -+ * iter->host, iter->addr, iter->rr and iter->mx are assigned concrete -+ * values, and while iter->port still corresponds to the nexthop service, -+ * or the default service configured with smtp_tcp_port or lmtp_tcp_port. -+ * -+ * When a connection is reused by nexthop/service or by server address/port, -+ * iter->host, iter->addr and iter->port are updated with actual values -+ * from the cached session. Additionally, when a connection is searched -+ * by nexthop/service, iter->rr remains null, and when a connection is -+ * searched by server address/port, iter->rr is updated with an actual -+ * server address/port before the search is made. -+ * - * First, search the cache by delivery request nexthop. We truncate the - * server address list when all the sessions for this destination are - * used up, to reduce the number of variables that need to be checked -@@ -757,9 +774,7 @@ static int smtp_reuse_session(SMTP_STATE *state, DNS_RR **addr_list, - /* XXX Assume there is no code at the end of this loop. */ - continue; - } -- vstring_strcpy(iter->addr, hostaddr.buf); -- vstring_strcpy(iter->host, SMTP_HNAME(addr)); -- iter->rr = addr; -+ SMTP_ITER_UPDATE_HOST(iter, SMTP_HNAME(addr), hostaddr.buf, addr); - #ifdef USE_TLS - if (!smtp_tls_policy_cache_query(why, state->tls, iter)) { - msg_warn("TLS policy lookup error for %s/%s: %s", -@@ -844,6 +859,7 @@ static void smtp_connect_inet(SMTP_STATE *state, const char *nexthop, - char *dest_buf; - char *domain; - unsigned port; -+ char *service; - DNS_RR *addr_list; - DNS_RR *addr; - DNS_RR *next; -@@ -851,6 +867,8 @@ static void smtp_connect_inet(SMTP_STATE *state, const char *nexthop, - int sess_count; - SMTP_SESSION *session; - int lookup_mx; -+ int non_dns_or_literal; -+ int i_am_mx; - unsigned domain_best_pref; - MAI_HOSTADDR_STR hostaddr; - -@@ -860,8 +878,28 @@ static void smtp_connect_inet(SMTP_STATE *state, const char *nexthop, - /* - * Parse the destination. If no TCP port is specified, use the port - * that is reserved for the protocol (SMTP or LMTP). -+ * -+ * The 'service' variable corresponds to the remote service specified -+ * with the nexthop, or the default service configured with -+ * smtp_tcp_port or lmtp_tcp_port. The 'port' variable and -+ * SMTP_ITERATOR.port initially correspond to that service. This -+ * determines what loop prevention will be in effect. -+ * -+ * The SMTP_ITERATOR.port will be overwritten after SRV record lookup. -+ * This guarantees that the connection cache key contains the correct -+ * port value when caching and retrieving a connection by its server -+ * address (and port). -+ * -+ * By design, the connection cache key contains NO port information when -+ * caching or retrieving a connection by its nexthop destination. -+ * Instead, the cache key contains the master.cf service name (a -+ * proxy for all the parameter settings including the default service -+ * from smtp_tcp_port or lmtp_tcp_port), together with the nexthop -+ * destination and sender-dependent info. This should be sufficient -+ * to avoid cross talk between mail streams that should be separated. - */ -- dest_buf = smtp_parse_destination(dest, def_service, &domain, &port); -+ dest_buf = smtp_parse_destination(dest, def_service, &domain, -+ &service, &port); - if (var_helpful_warnings && var_smtp_tls_wrappermode == 0 - && ntohs(port) == 465) { - msg_info("SMTPS wrappermode (TCP port 465) requires setting " -@@ -874,32 +912,48 @@ static void smtp_connect_inet(SMTP_STATE *state, const char *nexthop, - SMTP_ITER_INIT(iter, dest, NO_HOST, NO_ADDR, port, state); - - /* -- * Resolve an SMTP or LMTP server. In the case of SMTP, skip mail -- * exchanger lookups when a quoted host is specified or when DNS -- * lookups are disabled. -+ * Resolve an SMTP or LMTP server. Skip MX or SRV lookups when a -+ * quoted domain is specified or when DNS lookups are disabled. - */ - if (msg_verbose) -- msg_info("connecting to %s port %d", domain, ntohs(port)); -+ msg_info("connecting to %s service %s", domain, service); -+ non_dns_or_literal = (smtp_dns_support == SMTP_DNS_DISABLED -+ || *dest == '['); - if (smtp_mode) { - if (ntohs(port) == IPPORT_SMTP) - state->misc_flags |= SMTP_MISC_FLAG_LOOP_DETECT; - else - state->misc_flags &= ~SMTP_MISC_FLAG_LOOP_DETECT; -- lookup_mx = (smtp_dns_support != SMTP_DNS_DISABLED && *dest != '['); -+ lookup_mx = !non_dns_or_literal; - } else - lookup_mx = 0; -- if (!lookup_mx) { -+ -+ /* -+ * Look up SRV and address records and fall back to non-SRV lookups -+ * if permitted by configuration settings, or look up MX and address -+ * records, or look up address records only. -+ */ -+ i_am_mx = 0; -+ addr_list = 0; -+ if (!non_dns_or_literal && smtp_use_srv_lookup -+ && string_list_match(smtp_use_srv_lookup, service)) { -+ if (lookup_mx) -+ state->misc_flags |= SMTP_MISC_FLAG_FALLBACK_SRV_TO_MX; -+ else -+ state->misc_flags &= ~SMTP_MISC_FLAG_FALLBACK_SRV_TO_MX; -+ addr_list = smtp_service_addr(domain, service, &iter->mx, -+ state->misc_flags, why, &i_am_mx); -+ } else if (!lookup_mx) { -+ /* Non-DNS, literal, or non-SMTP service */ - addr_list = smtp_host_addr(domain, state->misc_flags, why); - /* XXX We could be an MX host for this destination... */ - } else { -- int i_am_mx = 0; -- - addr_list = smtp_domain_addr(domain, &iter->mx, state->misc_flags, - why, &i_am_mx); -- /* If we're MX host, don't connect to non-MX backups. */ -- if (i_am_mx) -- state->misc_flags |= SMTP_MISC_FLAG_FINAL_NEXTHOP; - } -+ /* If we're MX host, don't connect to non-MX backups. */ -+ if (i_am_mx) -+ state->misc_flags |= SMTP_MISC_FLAG_FINAL_NEXTHOP; - - /* - * Don't try fall-back hosts if mail loops to myself. That would just -@@ -992,9 +1046,7 @@ static void smtp_connect_inet(SMTP_STATE *state, const char *nexthop, - /* XXX Assume there is no code at the end of this loop. */ - continue; - } -- vstring_strcpy(iter->addr, hostaddr.buf); -- vstring_strcpy(iter->host, SMTP_HNAME(addr)); -- iter->rr = addr; -+ SMTP_ITER_UPDATE_HOST(iter, SMTP_HNAME(addr), hostaddr.buf, addr); - #ifdef USE_TLS - if (!smtp_tls_policy_cache_query(why, state->tls, iter)) { - msg_warn("TLS policy lookup for %s/%s: %s", -diff --git a/src/smtp/smtp_params.c b/src/smtp/smtp_params.c -index cd54f8f..22f4709 100644 ---- a/src/smtp/smtp_params.c -+++ b/src/smtp/smtp_params.c -@@ -66,6 +66,7 @@ - VAR_SMTP_DNS_RE_FILTER, DEF_SMTP_DNS_RE_FILTER, &var_smtp_dns_re_filter, 0, 0, - VAR_TLSPROXY_SERVICE, DEF_TLSPROXY_SERVICE, &var_tlsproxy_service, 1, 0, - VAR_HFROM_FORMAT, DEF_HFROM_FORMAT, &var_hfrom_format, 1, 0, -+ VAR_USE_SRV_LOOKUP, DEF_USE_SRV_LOOKUP, &var_use_srv_lookup, 0, 0, - 0, - }; - static const CONFIG_TIME_TABLE smtp_time_table[] = { -@@ -132,6 +133,8 @@ - VAR_SMTP_DUMMY_MAIL_AUTH, DEF_SMTP_DUMMY_MAIL_AUTH, &var_smtp_dummy_mail_auth, - VAR_SMTP_BALANCE_INET_PROTO, DEF_SMTP_BALANCE_INET_PROTO, &var_smtp_balance_inet_proto, - VAR_SMTP_BIND_ADDR_ENFORCE, DEF_SMTP_BIND_ADDR_ENFORCE, &var_smtp_bind_addr_enforce, -+ VAR_IGN_SRV_LOOKUP_ERR, DEF_IGN_SRV_LOOKUP_ERR, &var_ign_srv_lookup_err, -+ VAR_ALLOW_SRV_FALLBACK, DEF_ALLOW_SRV_FALLBACK, &var_allow_srv_fallback, - 0, - }; - static const CONFIG_NBOOL_TABLE smtp_nbool_table[] = { -diff --git a/src/smtp/smtp_session.c b/src/smtp/smtp_session.c -index 9f13978..90a0ff1 100644 ---- a/src/smtp/smtp_session.c -+++ b/src/smtp/smtp_session.c -@@ -129,6 +129,7 @@ - #define SESS_ATTR_DEST "destination" - #define SESS_ATTR_HOST "host_name" - #define SESS_ATTR_ADDR "host_addr" -+#define SESS_ATTR_PORT "host_port" - #define SESS_ATTR_DEST_FEATURES "destination_features" - - #define SESS_ATTR_TLS_LEVEL "tls_level" -@@ -259,6 +260,7 @@ int smtp_session_passivate(SMTP_SESSION *session, VSTRING *dest_prop, - SEND_ATTR_STR(SESS_ATTR_DEST, STR(iter->dest)), - SEND_ATTR_STR(SESS_ATTR_HOST, STR(iter->host)), - SEND_ATTR_STR(SESS_ATTR_ADDR, STR(iter->addr)), -+ SEND_ATTR_UINT(SESS_ATTR_PORT, iter->port), - SEND_ATTR_INT(SESS_ATTR_DEST_FEATURES, - session->features & SMTP_FEATURE_DESTINATION_MASK), - ATTR_TYPE_END) != 0 -@@ -398,9 +400,10 @@ SMTP_SESSION *smtp_session_activate(int fd, SMTP_ITERATOR *iter, - RECV_ATTR_STR(SESS_ATTR_DEST, iter->dest), - RECV_ATTR_STR(SESS_ATTR_HOST, iter->host), - RECV_ATTR_STR(SESS_ATTR_ADDR, iter->addr), -+ RECV_ATTR_UINT(SESS_ATTR_PORT, &iter->port), - RECV_ATTR_INT(SESS_ATTR_DEST_FEATURES, - &dest_features), -- ATTR_TYPE_END) != 4 -+ ATTR_TYPE_END) != 5 - || vstream_fclose(mp) != 0) { - msg_warn("smtp_session_passivate: bad cached dest properties"); - SMTP_SESSION_ACTIVATE_ERR_RETURN(); -diff --git a/src/smtpd/smtpd_check.c b/src/smtpd/smtpd_check.c -index a4a6af0..aa6200f 100644 ---- a/src/smtpd/smtpd_check.c -+++ b/src/smtpd/smtpd_check.c -@@ -3064,8 +3064,8 @@ static int check_server_access(SMTPD_STATE *state, const char *table, - || type == T_AAAA - #endif - ) { -- server_list = dns_rr_create(domain, domain, T_MX, C_IN, 0, 0, -- domain, strlen(domain) + 1); -+ server_list = dns_rr_create_nopref(domain, domain, T_MX, C_IN, 0, -+ domain, strlen(domain) + 1); - } else { - dns_status = dns_lookup(domain, type, 0, &server_list, - (VSTRING *) 0, (VSTRING *) 0); -@@ -3073,8 +3073,8 @@ static int check_server_access(SMTPD_STATE *state, const char *table, - return (SMTPD_CHECK_DUNNO); - if (dns_status == DNS_NOTFOUND /* Not: h_errno == NO_DATA */ ) { - if (type == T_MX) { -- server_list = dns_rr_create(domain, domain, type, C_IN, 0, 0, -- domain, strlen(domain) + 1); -+ server_list = dns_rr_create_nopref(domain, domain, type, C_IN, -+ 0, domain, strlen(domain) + 1); - dns_status = DNS_OK; - } else if (type == T_NS /* && h_errno == NO_DATA */ ) { - while ((domain = strchr(domain, '.')) != 0 && domain[1]) { -diff --git a/src/util/attr.h b/src/util/attr.h -index 067405f..7cd0cf2 100644 ---- a/src/util/attr.h -+++ b/src/util/attr.h -@@ -62,6 +62,7 @@ typedef int (*ATTR_PRINT_CUSTOM_FN) (ATTR_PRINT_COMMON_FN, VSTREAM *, int, const - * for documentation. - */ - #define SEND_ATTR_INT(name, val) ATTR_TYPE_INT, CHECK_CPTR(ATTR, char, (name)), CHECK_VAL(ATTR, int, (val)) -+#define SEND_ATTR_UINT(name, val) ATTR_TYPE_INT, CHECK_CPTR(ATTR, char, (name)), CHECK_VAL(ATTR, unsigned, (val)) - #define SEND_ATTR_STR(name, val) ATTR_TYPE_STR, CHECK_CPTR(ATTR, char, (name)), CHECK_CPTR(ATTR, char, (val)) - #define SEND_ATTR_HASH(val) ATTR_TYPE_HASH, CHECK_CPTR(ATTR, HTABLE, (val)) - #define SEND_ATTR_NV(val) ATTR_TYPE_NV, CHECK_CPTR(ATTR, NVTABLE, (val)) -@@ -70,6 +71,7 @@ typedef int (*ATTR_PRINT_CUSTOM_FN) (ATTR_PRINT_COMMON_FN, VSTREAM *, int, const - #define SEND_ATTR_FUNC(func, val) ATTR_TYPE_FUNC, CHECK_VAL(ATTR, ATTR_PRINT_CUSTOM_FN, (func)), CHECK_CPTR(ATTR, void, (val)) - - #define RECV_ATTR_INT(name, val) ATTR_TYPE_INT, CHECK_CPTR(ATTR, char, (name)), CHECK_PTR(ATTR, int, (val)) -+#define RECV_ATTR_UINT(name, val) ATTR_TYPE_INT, CHECK_CPTR(ATTR, char, (name)), CHECK_PTR(ATTR, unsigned, (val)) - #define RECV_ATTR_STR(name, val) ATTR_TYPE_STR, CHECK_CPTR(ATTR, char, (name)), CHECK_PTR(ATTR, VSTRING, (val)) - #define RECV_ATTR_STREQ(name, val) ATTR_TYPE_STREQ, CHECK_CPTR(ATTR, char, (name)), CHECK_CPTR(ATTR, char, (val)) - #define RECV_ATTR_HASH(val) ATTR_TYPE_HASH, CHECK_PTR(ATTR, HTABLE, (val)) -@@ -81,9 +83,11 @@ typedef int (*ATTR_PRINT_CUSTOM_FN) (ATTR_PRINT_COMMON_FN, VSTREAM *, int, const - CHECK_VAL_HELPER_DCL(ATTR, ssize_t); - CHECK_VAL_HELPER_DCL(ATTR, long); - CHECK_VAL_HELPER_DCL(ATTR, int); -+CHECK_VAL_HELPER_DCL(ATTR, unsigned); - CHECK_PTR_HELPER_DCL(ATTR, void); - CHECK_PTR_HELPER_DCL(ATTR, long); - CHECK_PTR_HELPER_DCL(ATTR, int); -+CHECK_PTR_HELPER_DCL(ATTR, unsigned); - CHECK_PTR_HELPER_DCL(ATTR, VSTRING); - CHECK_PTR_HELPER_DCL(ATTR, NVTABLE); - CHECK_PTR_HELPER_DCL(ATTR, HTABLE); diff --git a/postfix-3.7.0-config.patch b/postfix-3.8.0-config.patch similarity index 97% rename from postfix-3.7.0-config.patch rename to postfix-3.8.0-config.patch index d8be512fdb2661e524dfca17c07633cf384e4cb0..f77cdc382f8143b7f136eb973253b42430628212 100644 --- a/postfix-3.7.0-config.patch +++ b/postfix-3.8.0-config.patch @@ -1,5 +1,5 @@ diff --git a/conf/main.cf b/conf/main.cf -index 47de434..112c1f1 100644 +index 2ee7996..336bd7b 100644 --- a/conf/main.cf +++ b/conf/main.cf @@ -136,6 +136,10 @@ mail_owner = postfix @@ -123,10 +123,10 @@ index 47de434..112c1f1 100644 +# +smtp_tls_security_level = may diff --git a/conf/master.cf b/conf/master.cf -index 83fc6fd..bb0eae9 100644 +index fd282dd..8d969c6 100644 --- a/conf/master.cf +++ b/conf/master.cf -@@ -110,14 +110,14 @@ postlog unix-dgram n - n - 1 postlogd +@@ -112,14 +112,14 @@ postlog unix-dgram n - n - 1 postlogd # Also specify in main.cf: cyrus_destination_recipient_limit=1 # #cyrus unix - n n - - pipe diff --git a/postfix-3.7.9-large-fs.patch b/postfix-3.8.0-large-fs.patch similarity index 92% rename from postfix-3.7.9-large-fs.patch rename to postfix-3.8.0-large-fs.patch index 28c1eabe0194ad958340554737123754ccc87947..b0c7933316720a410a7d473f5f635fce235494c1 100644 --- a/postfix-3.7.9-large-fs.patch +++ b/postfix-3.8.0-large-fs.patch @@ -21,10 +21,10 @@ index 50a4aa7..beef3db 100644 if (msg_verbose) msg_info("%s: %s: block size %lu, blocks free %lu", diff --git a/src/util/sys_defs.h b/src/util/sys_defs.h -index aec8d8a..aff8d44 100644 +index 1fb449d..bcaac27 100644 --- a/src/util/sys_defs.h +++ b/src/util/sys_defs.h -@@ -772,8 +772,8 @@ extern int initgroups(const char *, int); +@@ -783,8 +783,8 @@ extern int initgroups(const char *, int); #define GETTIMEOFDAY(t) gettimeofday(t,(struct timezone *) 0) #define ROOT_PATH "/bin:/usr/bin:/sbin:/usr/sbin" #define FIONREAD_IN_TERMIOS_H diff --git a/postfix-3.8.5-openssl-no-engine.patch b/postfix-3.8.5-openssl-no-engine.patch new file mode 100644 index 0000000000000000000000000000000000000000..8efb0e62b6af762161e9087bd4b2283175ed1c03 --- /dev/null +++ b/postfix-3.8.5-openssl-no-engine.patch @@ -0,0 +1,12 @@ +diff --git a/src/posttls-finger/posttls-finger.c b/src/posttls-finger/posttls-finger.c +index b9a4699..29e6ec1 100644 +--- a/src/posttls-finger/posttls-finger.c ++++ b/src/posttls-finger/posttls-finger.c +@@ -405,7 +405,6 @@ + + #ifdef USE_TLS + #include +-#include + #endif + + /* diff --git a/postfix-3.7.9.tar.gz b/postfix-3.8.5.tar.gz similarity index 40% rename from postfix-3.7.9.tar.gz rename to postfix-3.8.5.tar.gz index a57f7190b500eca74cf8e9a526c95ae164108f7c..0b4f221a831ecb2c55d19e0f952073825cd6d601 100644 Binary files a/postfix-3.7.9.tar.gz and b/postfix-3.8.5.tar.gz differ diff --git a/postfix.spec b/postfix.spec index cf68c3a34c00b7fac9c7139d50438c992740e4d6..f31ee31c70343ac49f4be3211f51faa2418f7b4a 100644 --- a/postfix.spec +++ b/postfix.spec @@ -15,6 +15,12 @@ %bcond_without ipv6 %bcond_without pflogsumm +%if %{with lmdb} +%global defmap_lmdb 1 +%else +%global defmap_lmdb 0 +%endif + %global sysv2systemdnvr 2.8.12-2 # hardened build if not overrided @@ -23,12 +29,9 @@ # Postfix requires one exlusive uid/gid and a 2nd exclusive gid for its own # use. Let me know if the second gid collides with another package. # Be careful: Redhat's 'mail' user & group isn't unique! -%define postfix_uid 89 +# It's now handled by systemd-sysusers. %define postfix_user postfix -%define postfix_gid 89 -%define postfix_group postfix %define maildrop_group postdrop -%define maildrop_gid 90 %define postfix_config_dir %{_sysconfdir}/postfix %define postfix_daemon_dir %{_libexecdir}/postfix @@ -50,11 +53,11 @@ Name: postfix Summary: Postfix Mail Transport Agent -Version: 3.7.9 +Version: 3.8.5 Release: %{anolis_release}%{?dist} Epoch: 2 URL: http://www.postfix.org -License: (IBM and GPLv2+) or (EPL-2.0 and GPLv2+) +License: (IPL-1.0 OR EPL-2.0) AND GPL-2.0-or-later AND BSD-4-Clause-UC Requires(post): systemd systemd-sysv hostname Requires(post): %{_sbindir}/alternatives Requires(post): %{_bindir}/openssl @@ -76,6 +79,7 @@ Source2: postfix.service Source3: README-Postfix-SASL-RedHat.txt Source4: postfix.aliasesdb Source5: postfix-chroot-update +Source6: postfix.sysusers # Sources 50-99 are upstream [patch] contributions @@ -91,31 +95,29 @@ Source101: postfix-pam.conf # Patches -Patch1: postfix-3.7.0-config.patch +Patch1: postfix-3.8.0-config.patch Patch2: postfix-3.4.0-files.patch Patch3: postfix-3.3.3-alternatives.patch -Patch4: postfix-3.7.9-large-fs.patch +Patch4: postfix-3.8.0-large-fs.patch Patch9: pflogsumm-1.1.5-datecalc.patch # rhbz#1384871, sent upstream Patch10: pflogsumm-1.1.5-ipv6-warnings-fix.patch Patch11: postfix-3.4.4-chroot-example-fix.patch -# sent upstream -Patch12: postfix-3.7.0-whitespace-name-fix.patch # rhbz#1931403, sent upstream Patch13: pflogsumm-1.1.5-syslog-name-underscore-fix.patch -# backported feature from upstream -Patch14: postfix-3.7.9-SRV-resolve.patch +Patch14: postfix-3.8.5-openssl-no-engine.patch # Optional patches - set the appropriate environment variables to include # them when building the package/spec file # Determine the different packages required for building postfix -BuildRequires: make +BuildRequires: make systemd-rpm-macros sed BuildRequires: libdb-devel, perl-generators, pkgconfig, zlib-devel BuildRequires: systemd-units, libicu-devel BuildRequires: gcc, m4, findutils +%{?with_db:BuildRequires: libdb-devel} %{?with_ldap:BuildRequires: openldap-devel} %{?with_lmdb:BuildRequires: lmdb-devel} %{?with_sasl:BuildRequires: cyrus-sasl-devel} @@ -126,6 +128,10 @@ BuildRequires: gcc, m4, findutils %{?with_cdb:BuildRequires: tinycdb-devel} %{?with_tls:BuildRequires: openssl-devel} +%if 0%{?defmap_lmdb} +Requires: %{name}-lmdb = %{EVR} +%endif + %description Postfix is a Mail Transport Agent (MTA). @@ -240,9 +246,8 @@ pushd pflogsumm-%{pflogsumm_ver} popd %endif %patch11 -p1 -b .chroot-example-fix -%patch12 -p1 -b .whitespace-name-fix %patch13 -p1 -b .pflogsumm-1.1.5-syslog-name-underscore-fix -%patch14 -p1 -b .SRV +%patch14 -p1 -b .openssl-no-engine # Backport 3.8-20221006 fix for uname -r detection sed -i makedefs -e '\@Linux\.@s|345|3456|' @@ -253,12 +258,24 @@ for f in README_FILES/TLS_{LEGACY_,}README TLS_ACKNOWLEDGEMENTS; do touch -r ${f}{,_} && mv -f ${f}{_,} done +# fix default maps +%if 0%{?defmap_lmdb} + sed -i '/^\s*alias_maps\s*=\s*hash:\/etc\/aliases/ s|hash:|lmdb:|g' conf/main.cf + sed -i '/^\s*alias_database\s*=\s*hash:\/etc\/aliases/ s|hash:|lmdb:|g' conf/main.cf + echo >> conf/main.cf + echo "default_database_type = lmdb" >> conf/main.cf +%endif + %build %set_build_flags unset AUXLIBS AUXLIBS_LDAP AUXLIBS_LMDB AUXLIBS_PCRE AUXLIBS_MYSQL AUXLIBS_PGSQL AUXLIBS_SQLITE AUXLIBS_CDB CCARGS="-fPIC -fcommon" AUXLIBS="" +%if %{without db} + CCARGS="${CCARGS} -DNO_DB" +%endif + %if %{with ldap} CCARGS="${CCARGS} -DHAS_LDAP -DLDAP_DEPRECATED=1 %{?with_sasl:-DUSE_LDAP_SASL}" AUXLIBS_LDAP="-lldap -llber" @@ -360,6 +377,9 @@ install -m 644 %{SOURCE2} %{buildroot}%{_unitdir} install -m 755 %{SOURCE4} %{buildroot}%{postfix_daemon_dir}/aliasesdb install -m 755 %{SOURCE5} %{buildroot}%{postfix_daemon_dir}/chroot-update +# systemd-sysusers +install -p -D -m 0644 %{SOURCE6} %{buildroot}%{_sysusersdir}/postfix.conf + install -c auxiliary/rmail/rmail $RPM_BUILD_ROOT%{_bindir}/rmail.postfix for i in active bounce corrupt defer deferred flush incoming private saved maildrop public pid saved trace; do @@ -515,10 +535,7 @@ exit 0 %pre # Add user and groups if necessary -%{_sbindir}/groupadd -g %{maildrop_gid} -r %{maildrop_group} 2>/dev/null -%{_sbindir}/groupadd -g %{postfix_gid} -r %{postfix_group} 2>/dev/null -%{_sbindir}/groupadd -g 12 -r mail 2>/dev/null -%{_sbindir}/useradd -d %{postfix_queue_dir} -s /sbin/nologin -g %{postfix_group} -G mail -M -r -u %{postfix_uid} %{postfix_user} 2>/dev/null +%sysusers_create_compat %{SOURCE6} # hack, to turn man8/smtpd.8.gz into alternatives symlink (part of the rhbz#1051180 fix) # this could be probably dropped in f23+ @@ -632,7 +649,7 @@ exit 0 %attr(0755, root, root) %{postfix_command_dir}/postfix %attr(0755, root, root) %{postfix_command_dir}/postkick %attr(0755, root, root) %{postfix_command_dir}/postlock -%attr(0755, root, root) %{postfix_command_dir}/postlog +%attr(2755, root, %{maildrop_group}) %{postfix_command_dir}/postlog %attr(0755, root, root) %{postfix_command_dir}/postmap %attr(0755, root, root) %{postfix_command_dir}/postmulti %attr(2755, root, %{maildrop_group}) %{postfix_command_dir}/postqueue @@ -684,6 +701,8 @@ exit 0 %ghost %attr(0755, root, root) %{_prefix}/lib/sendmail %ghost %attr(0644, root, root) %{_var}/lib/misc/postfix.aliasesdb-stamp +# systemd-sysusers +%{_sysusersdir}/postfix.conf %files perl-scripts %attr(0755, root, root) %{postfix_command_dir}/qshape @@ -758,6 +777,15 @@ exit 0 %endif %changelog +* Wed Mar 19 2025 happy_orange - 2:3.8.5-1 +- update to 3.8.5 +- Explicitly set default_database_type if lmdb map is used +- Converted license tag to SPDX +- Use systemd-sysusers, original patch by Jonathan Wright +- Dropped mail group configuration / creation, it is provided by setup + [Patches cherry-pick from c10s. Original changelog: + - Dropped mail group configuration / creation, it is provided by setup.(3.8.5-7)] + * Wed Jan 10 2024 mgb01105731 - 2:3.7.9-1 - New version 3.7.9 diff --git a/postfix.sysusers b/postfix.sysusers new file mode 100644 index 0000000000000000000000000000000000000000..7778edf7f936b9bb1d604f7b0919e776b9309935 --- /dev/null +++ b/postfix.sysusers @@ -0,0 +1,3 @@ +u postfix 89 - /var/spool/postfix /sbin/nologin +g postdrop 90 +m postfix mail