net.java.truelicense.core.BasicLicenseApplicationContext Maven / Gradle / Ivy
/*
* Copyright (C) 2005-2015 Schlichtherle IT Services.
* All rights reserved. Use is subject to license terms.
*/
package net.java.truelicense.core;
import net.java.truelicense.core.auth.Authentication;
import net.java.truelicense.core.auth.KeyStoreParameters;
import net.java.truelicense.core.auth.Repository;
import net.java.truelicense.core.auth.RepositoryProvider;
import net.java.truelicense.core.codec.Codec;
import net.java.truelicense.core.crypto.Encryption;
import net.java.truelicense.core.crypto.PbeParameters;
import net.java.truelicense.core.io.*;
import net.java.truelicense.core.policy.PasswordPolicy;
import net.java.truelicense.core.policy.PasswordPolicyProvider;
import net.java.truelicense.core.policy.WeakPasswordException;
import net.java.truelicense.core.util.ClassLoaderProvider;
import net.java.truelicense.core.util.Clock;
import net.java.truelicense.obfuscate.ObfuscatedString;
import javax.annotation.CheckForNull;
import javax.annotation.concurrent.Immutable;
import java.io.File;
import java.util.Arrays;
import java.util.Calendar;
import java.util.Date;
import java.util.prefs.Preferences;
import static java.util.Calendar.DATE;
import static java.util.Calendar.getInstance;
import static net.java.truelicense.core.util.Objects.requireNonNull;
/**
* A basic context for license applications.
*
* Unless stated otherwise, all no-argument methods need to return consistent
* objects so that caching them is not required.
* A returned object is considered to be consistent if it compares
* {@linkplain Object#equals(Object) equal} or at least behaves identical to
* any previously returned object.
*
* @author Christian Schlichtherle
*/
@Immutable
abstract class BasicLicenseApplicationContext
implements ClassLoaderProvider,
Clock,
LicenseApplicationContext,
LicenseSubjectProvider,
PasswordPolicyProvider {
private final LicenseManagementContext context;
BasicLicenseApplicationContext(final LicenseManagementContext context) {
assert null != context;
this.context = context;
}
@Override public final String subject() { return context().subject(); }
@Override public final LicenseManagementContext context() {
return context;
}
@Override public final PasswordPolicy policy() {
return context().policy();
}
@Override public final Date now() { return context().now(); }
@Override public final @CheckForNull ClassLoader classLoader() {
return context().classLoader();
}
final LicenseParameters parameters(
Authentication authentication,
Encryption encryption) {
return parameters(context().initialization(), authentication, encryption);
}
final LicenseParameters ftpParameters(
final LicenseConsumerManager parent,
final Authentication authentication,
final @CheckForNull Encryption encryption,
final int days) {
if (0 >= days) throw new IllegalArgumentException();
final LicenseInitialization initialization = new LicenseInitialization() {
final LicenseInitialization initialization = context().initialization();
@Override public void initialize(final License bean) {
initialization.initialize(bean);
final Calendar cal = getInstance();
cal.setTime(bean.getIssued());
bean.setNotBefore(cal.getTime()); // not before issued
cal.add(DATE, days); // FTP countdown starts NOW
bean.setNotAfter(cal.getTime());
}
};
return chainedParameters(parent, initialization, authentication, encryption);
}
final LicenseParameters chainedParameters(
final LicenseConsumerManager parent,
final Authentication authentication,
final @CheckForNull Encryption encryption) {
return chainedParameters(parent, context().initialization(),
authentication, encryption);
}
private LicenseParameters chainedParameters(
LicenseConsumerManager parent,
LicenseInitialization initialization,
Authentication authentication,
@CheckForNull Encryption encryption) {
return parameters(initialization, authentication,
resolveEncryption(parent, encryption));
}
private static Encryption resolveEncryption(
LicenseConsumerManager parent,
@CheckForNull Encryption encryption) {
return null != encryption ? encryption : parent.parameters().encryption();
}
private LicenseParameters parameters(
final LicenseInitialization initialization,
final Authentication authentication,
final Encryption encryption) {
final LicenseManagementContext c = context();
return parameters(c.authorization(), initialization, c.validation(),
c, authentication, c.codec(), c.compression(), encryption);
}
private static LicenseParameters parameters(
final LicenseAuthorization authorization,
final LicenseInitialization initialization,
final LicenseValidation validation,
final RepositoryProvider rp,
final Authentication authentication,
final Codec codec,
final Transformation compression,
final Encryption encryption) {
requireNonNull(authorization);
requireNonNull(initialization);
requireNonNull(validation);
requireNonNull(authentication);
requireNonNull(codec);
requireNonNull(compression);
requireNonNull(encryption);
return new LicenseParameters() {
@Override public LicenseAuthorization authorization() { return authorization; }
@Override public LicenseInitialization initialization() { return initialization; }
@Override public LicenseValidation validation() { return validation; }
@Override public Repository repository() { return rp.repository(); }
@Override public Authentication authentication() { return authentication; }
@Override public Codec codec() { return codec; }
@Override public Transformation compression() { return compression; }
@Override public Encryption encryption() { return encryption; }
};
}
final KeyStoreParameters kspChecked(
final @CheckForNull Source source,
final @CheckForNull String storeType,
final ObfuscatedString storePassword,
final String alias,
final @CheckForNull ObfuscatedString keyPassword) {
final KeyStoreParameters ksp = kspUnchecked(source, storeType,
storePassword, alias, keyPassword);
check(ksp.storePassword());
check(ksp.keyPassword());
return ksp;
}
@SuppressWarnings("deprecation")
final KeyStoreParameters kspUnchecked(
final @CheckForNull Source source,
final @CheckForNull String storeType,
final ObfuscatedString storePassword,
final String alias,
final @CheckForNull ObfuscatedString keyPassword) {
requireNonNull(storePassword);
requireNonNull(alias);
return new KeyStoreParameters() {
@Override public @CheckForNull Source source() { return source; }
@Override public String storeType() {
return null != storeType
? storeType
: context().storeType();
}
@Override public char[] storePassword() {
return storePassword.toCharArray();
}
@Override public String alias() { return alias; }
@Override public char[] keyPassword() {
final char[] pwd = keyPassword0();
return 0 != pwd.length ? pwd : storePassword();
}
@Override public boolean forSigning() {
return 0 != keyPassword0().length;
}
char[] keyPassword0() {
return null != keyPassword
? keyPassword.toCharArray()
: new char[0];
}
};
}
final PbeParameters pbeChecked(
final @CheckForNull String algorithm,
final ObfuscatedString password) {
final PbeParameters pp = pbeUnchecked(algorithm, password);
check(pp.password());
return pp;
}
final PbeParameters pbeUnchecked(
final @CheckForNull String algorithm,
final ObfuscatedString password) {
requireNonNull(password);
return new PbeParameters() {
final String resolvedAlgorithm = resolvePbeAlgorithm(algorithm);
@Override public String algorithm() { return resolvedAlgorithm; }
@Override public char[] password() { return password.toCharArray(); }
};
}
private String resolvePbeAlgorithm(@CheckForNull String algorithm) {
return null != algorithm ? algorithm : context().pbeAlgorithm();
}
private void check(final char[] password) {
try {
policy().check(password);
} catch (final WeakPasswordException ex) {
throw new IllegalArgumentException(ex);
} finally {
Arrays.fill(password, (char) 0);
}
}
@Override public Source resource(String name) {
return Sources.forResource(name, classLoader());
}
@Override public Store systemNodeStore(Class> classInPackage) {
return new PreferencesStore(
Preferences.systemNodeForPackage(classInPackage), subject());
}
@Override public Store userNodeStore(Class> classInPackage) {
return new PreferencesStore(
Preferences.userNodeForPackage(classInPackage), subject());
}
@Override public Store fileStore(File file) { return new FileStore(file); }
}