G'day,

I have a Postfix+Dovecot+LDAP set up with multi-level sub-domains. I had no problem setting Dovecot up for this environment with the '%D' config variable modifier.

 * 
https://doc.dovecot.org/configuration_manual/config_file/config_variables/#modifiers
 * 
https://github.com/dovecot/core/blob/6ec7c04b43b3a31dc9dcf57d1bdeec121f81d13e/src/lib/var-expand.c#L129

It's a simple modifier that substitutes all the dots in the string to ",dc=".

For the sake of explanation, here's an example of the multi-level subdomains:

   @startwbs
   * dc=example,dc=com
   ** dc=admin,dc=example,dc=com
   *** dc=hr,dc=admin,dc=example,dc=com
   *** dc=accounting,dc=admin,dc=example,dc=com
   ** dc=legal,dc=example,dc=com
   ** dc=dev,dc=example,dc=com
   *** dc=this-proejct,dc=dev,dc=example,dc=com
   *** dc=that-project,dc=dev,dc=example,dc=com
   ** dc=marketing,dc=example,dc=com
   *** dc=domestic,dc=marketing,dc=example,dc=com
   *** dc=overseas,dc=marketing,dc=example,dc=com
   @endwbs

I fail to see the way to have just one set of ldap_table cf config files for the Postfix set up. I know I could use %[1-9] modifier to configure subdomains up to 9 levels from the root. But this means I'd have to manage 9 different sets of cf files.

   search_base = dc=%1,dc=%2,dc=%3 ...


With '%l' expansion I propose, I can have just one set of cf files to serve all the subdomains. Here's an example:

   # Use the regexp config to restrict the use of this ldap cf to certain 
domains
   domain = regexp:domain-regexp.cf

   # Connection info
   version = 3
   server_host = ldapi:///
   bind = sasl
   sasl_mechs = EXTERNAL

   *search_base = dc=%l*
   query_filter = (&(objectClass=x-your-mailbox-class)(mail=%s))
   result_attribute = mail

In domain-regexp.cf:

   # Match the root domain and its subdomains. The "OK" is just a placeholder as
   # this regexp_table is not used for mapping.
   /^(([a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?\.)+)?example\.com$/ OK

With these, you can serve example.com and all of its subdomains as long as the directory follows the same "dc=" structure. You can use the regexp_table to filter out certain subdomains if you'd like. (domain= is such a neat filter. I wish Dovecot had something similar for userdb/passdb driver)


Attached is the draft implementation of '%l'. I couldn't figure out how to fit this in properly as I'm not familiar with the structure of _expand() and _parse(). I implemented it in db_common_parse() and db_common_expand(), but I don't like this approach since the '%l' is only relevant to ldap_table. But implementing separate functions just for ldap_table is way above my payroll, so I just decided to implement it in those two functions.

And here's the wording for proto docs.

   # .IP "\fB%l\fR"
   #    When the input key is an address of the form user@domain, \fB%l\fR
   #    is replaced by the (RFC 4514) LDAP DN representation of the domain
   #    part of the address using the (RFC 4519) 'dc' attribute. The value
   #    of each 'dc' attribute is (QUOTE_STYLE) quoted. For example, the
   #    input key"u...@sub.domain.org"  is replaced by "sub,dc=domain,dc=org".
   #    Otherwise, the search is suppressed and returns no results.


Regards,

Davo
diff --git a/postfix/src/global/db_common.c b/postfix/src/global/db_common.c
index 15e7a1c1..deb6b89d 100644
--- a/postfix/src/global/db_common.c
+++ b/postfix/src/global/db_common.c
@@ -207,6 +207,11 @@ int     db_common_parse(DICT *dict, void **ctxPtr, const char *format, int query
 		    : DB_COMMON_VALUE_USER;
 		dynamic = 1;
 		break;
+		case 'l':
+		 /* Always use parts to compose DN */
+		if (ctx->nparts == 0)
+		    ctx->nparts = 1;
+		/* FALLTHROUGH */
 	    case 'd':
 		ctx->flags |=
 		    query ? DB_COMMON_KEY_DOMAIN | DB_COMMON_KEY_PARTIAL
@@ -399,6 +404,13 @@ int     db_common_expand(void *ctxArg, const char *format, const char *value,
 	else \
 	    vstring_strcat(buf, v); \
     } while (0)
+#define PANIC_DFLAG() \
+	if (!(ctx->flags & dflag)) \
+		msg_panic("%s: %s: %s: bad query/result template context", \
+				myname, ctx->dict->name, format); \
+	if (!vdomain) \
+		msg_panic("%s: %s: %s: expanding domain-less key or value", \
+				myname, ctx->dict->name, format);
 
     /*
      * Replace all instances of %s with the address to look up. Replace %u
@@ -427,15 +439,19 @@ int     db_common_expand(void *ctxArg, const char *format, const char *value,
 		break;
 
 	    case 'd':
-		if (!(ctx->flags & dflag))
-		    msg_panic("%s: %s: %s: bad query/result template context",
-			      myname, ctx->dict->name, format);
-		if (!vdomain)
-		    msg_panic("%s: %s: %s: expanding domain-less key or value",
-			      myname, ctx->dict->name, format);
+		PANIC_DFLAG();
 		QUOTE_VAL(ctx->dict, quote_func, vdomain, result);
 		break;
 
+		case 'l':
+		PANIC_DFLAG();
+		QUOTE_VAL(ctx->dict, quote_func, parts->argv[0], result);
+		for (i = 1; i < parts->argc; i++) {
+			vstring_strcat(result, ",dc=");
+			QUOTE_VAL(ctx->dict, quote_func, parts->argv[i], result);
+		}
+		break;
+
 	    case 'S':
 		if (key)
 		    QUOTE_VAL(ctx->dict, quote_func, key, result);
@@ -518,6 +534,8 @@ int     db_common_expand(void *ctxArg, const char *format, const char *value,
 	argv_free(parts);
 
     return (1);
+#undef QUOTE_VAL
+#undef PANIC_DFLAG
 }
 
 

Reply via email to