/*
|
* The contents of this file are subject to the terms of the Common Development and
|
* Distribution License (the License). You may not use this file except in compliance with the
|
* License.
|
*
|
* You can obtain a copy of the License at legal/CDDLv1.0.txt. See the License for the
|
* specific language governing permission and limitations under the License.
|
*
|
* When distributing Covered Software, include this CDDL Header Notice in each file and include
|
* the License file at legal/CDDLv1.0.txt. If applicable, add the following below the CDDL
|
* Header, with the fields enclosed by brackets [] replaced by your own identifying
|
* information: "Portions copyright [year] [name of copyright owner]".
|
*
|
* Copyright 2013 ForgeRock AS.
|
*/
|
|
package org.forgerock.opendj.rest2ldap;
|
|
import static org.forgerock.opendj.rest2ldap.Utils.i18n;
|
import static org.forgerock.opendj.rest2ldap.Utils.isJSONPrimitive;
|
|
import java.util.ArrayList;
|
import java.util.List;
|
import java.util.Locale;
|
import java.util.Map;
|
import java.util.regex.Matcher;
|
import java.util.regex.Pattern;
|
|
import org.forgerock.json.resource.ForbiddenException;
|
import org.forgerock.json.resource.ResourceException;
|
import org.forgerock.opendj.ldap.DN;
|
import org.forgerock.opendj.ldap.schema.Schema;
|
|
/**
|
* An authorization ID template used for mapping security context principals to
|
* AuthzID templates of the form
|
* <code>dn:uid={uid},ou={realm},dc=example,dc=com</code>, or
|
* <code>u:{uid}@{realm}.example.com</code>.
|
*/
|
final class AuthzIdTemplate {
|
private static interface Impl {
|
String formatAsAuthzId(AuthzIdTemplate t, Object[] templateVariables, Schema schema)
|
throws ResourceException;
|
}
|
|
private static final Impl DN_IMPL = new Impl() {
|
|
@Override
|
public String formatAsAuthzId(final AuthzIdTemplate t, final Object[] templateVariables,
|
final Schema schema) throws ResourceException {
|
final String authzId = String.format(Locale.ENGLISH, t.formatString, templateVariables);
|
try {
|
// Validate the DN.
|
DN.valueOf(authzId.substring(3), schema);
|
} catch (final IllegalArgumentException e) {
|
throw new ForbiddenException(
|
i18n("The request could not be authorized because the required "
|
+ "security principal was not a valid LDAP DN"));
|
}
|
return authzId;
|
}
|
};
|
|
private static final Pattern DN_PATTERN = Pattern.compile("^dn:\\{[^}]+\\}$");
|
|
private static final Impl DN_TEMPLATE_IMPL = new Impl() {
|
|
@Override
|
public String formatAsAuthzId(final AuthzIdTemplate t, final Object[] templateVariables,
|
final Schema schema) throws ResourceException {
|
return "dn:" + DN.format(t.dnFormatString, schema, templateVariables).toString();
|
}
|
|
};
|
|
private static final Pattern KEY_RE = Pattern.compile("\\{([^}]+)\\}");
|
|
private static final Impl UID_TEMPLATE_IMPL = new Impl() {
|
|
@Override
|
public String formatAsAuthzId(final AuthzIdTemplate t, final Object[] templateVariables,
|
final Schema schema) throws ResourceException {
|
return String.format(Locale.ENGLISH, t.formatString, templateVariables);
|
}
|
|
};
|
|
private final String dnFormatString;
|
private final String formatString;
|
private final List<String> keys = new ArrayList<String>();
|
private final Impl pimpl;
|
private final String template;
|
|
AuthzIdTemplate(final String template) {
|
if (!template.startsWith("u:") && !template.startsWith("dn:")) {
|
throw new IllegalArgumentException("Invalid authorization ID template: " + template);
|
}
|
|
// Parse the template keys and replace them with %s for formatting.
|
final Matcher matcher = KEY_RE.matcher(template);
|
final StringBuffer buffer = new StringBuffer(template.length());
|
while (matcher.find()) {
|
matcher.appendReplacement(buffer, "%s");
|
keys.add(matcher.group(1));
|
}
|
matcher.appendTail(buffer);
|
this.formatString = buffer.toString();
|
this.template = template;
|
|
if (template.startsWith("dn:")) {
|
this.pimpl = DN_PATTERN.matcher(template).matches() ? DN_IMPL : DN_TEMPLATE_IMPL;
|
this.dnFormatString = formatString.substring(3);
|
} else {
|
this.pimpl = UID_TEMPLATE_IMPL;
|
this.dnFormatString = null;
|
}
|
}
|
|
@Override
|
public String toString() {
|
return template;
|
}
|
|
String formatAsAuthzId(final Map<String, Object> principals, final Schema schema)
|
throws ResourceException {
|
final String[] templateVariables = getPrincipalsForFormatting(principals);
|
return pimpl.formatAsAuthzId(this, templateVariables, schema);
|
}
|
|
private String[] getPrincipalsForFormatting(final Map<String, Object> principals)
|
throws ForbiddenException {
|
final String[] values = new String[keys.size()];
|
for (int i = 0; i < values.length; i++) {
|
final String key = keys.get(i);
|
final Object value = principals.get(key);
|
if (isJSONPrimitive(value)) {
|
values[i] = String.valueOf(value);
|
} else if (value == null) {
|
throw new ForbiddenException(i18n(
|
"The request could not be authorized because the required "
|
+ "security principal '%s' could not be determined", key));
|
} else {
|
throw new ForbiddenException(i18n(
|
"The request could not be authorized because the required "
|
+ "security principal '%s' had an invalid data type", key));
|
}
|
}
|
return values;
|
}
|
}
|