com.authlete.common.api.AuthleteApiFactory Maven / Gradle / Ivy
Show all versions of authlete-java-common Show documentation
/*
* Copyright (C) 2014-2022 Authlete, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.authlete.common.api;
import java.lang.reflect.Constructor;
import com.authlete.common.conf.AuthleteApiVersion;
import com.authlete.common.conf.AuthleteConfiguration;
import com.authlete.common.conf.AuthletePropertiesConfiguration;
/**
* Factory to create an {@link AuthleteApi} instance.
*
* @author Takahiko Kawasaki
*/
public class AuthleteApiFactory
{
/**
* An implementation of {@link AuthleteApi} using JAX-RS.
* This implementation exists in authlete/authlete-java-jaxrs.
*/
private static final String IMPL_JAX_RS_V2 = "com.authlete.jaxrs.api.AuthleteApiImpl";
/**
* An implementation of {@link AuthleteApi} using JAX-RS.
* This implementation exists in authlete/authlete-java-jaxrs.
*/
private static final String IMPL_JAX_RS_V3 = "com.authlete.jaxrs.api.AuthleteApiImplV3";
/**
* An implementation of {@link AuthleteApi} using JAKARTA.
* This implementation exists in authlete/authlete-java-jakarta.
*/
private static final String IMPL_JAKARTA_V2 = "com.authlete.jakarta.api.AuthleteApiImpl";
/**
* An implementation of {@link AuthleteApi} using JAKARTA.
* This implementation exists in authlete/authlete-java-jakarta.
*/
private static final String IMPL_JAKARTA_V3 = "com.authlete.jakarta.api.AuthleteApiImplV3";
/**
* An implementation of {@link AuthleteApi} using {@link java.net.HttpURLConnection HttpURLConnection}.
* This implementation exists in authlete/authlete-java-common.
*/
private static final String IMPL_HTTP_URL_CONNECTION = "com.authlete.common.api.AuthleteApiImpl";
/**
* Known implementations of AuthleteApi V2.
*/
private static final String[] sKnownImplsV2 = {
IMPL_JAX_RS_V2,
IMPL_JAKARTA_V2,
IMPL_HTTP_URL_CONNECTION
};
/**
* Known implementations of AuthleteApi V3.
*/
private static final String[] sKnownImplsV3 = {
IMPL_JAX_RS_V3,
IMPL_JAKARTA_V3
};
/**
* The default {@link AuthleteApi} instance.
*/
private static volatile AuthleteApi sDefaultApi;
private AuthleteApiFactory()
{
}
/**
* Create an instance of {@link AuthleteApi}.
*
*
* This method repeats to call {@link #create(AuthleteConfiguration, String)}
* until one of the known classes is successfully instantiated.
*
*
*
* The classes listed below are the ones that the current implementation knows
* for Authlete API V3.
*
*
*
* com.authlete.jaxrs.api.AuthleteApiImplV3
* (using JAX-RS 2.0 API, contained in com.authlete:authlete-java-jaxrs
)
*
*
*
* The classes listed below are the ones that the current implementation knows
* for Authlete API V2.
*
*
*
* com.authlete.jaxrs.api.AuthleteApiImpl
* (using JAX-RS 2.0 API, contained in com.authlete:authlete-java-jaxrs
)
* com.authlete.common.api.AuthleteApiImpl
* (using {@link java.net.HttpURLConnection HttpURLConnection}, contained in com.authlete:authlete-java-common
since version 2.0)
*
*
* @param configuration
* Authlete configuration.
*
* @return
* An instance of {@link AuthleteApi}. If none of the known classes
* that implement {@code AuthleteApi} interface was successfully
* instantiated, {@code null} is returned.
*/
public static AuthleteApi create(AuthleteConfiguration configuration)
{
// Authlete API version specified by the configuration.
AuthleteApiVersion version =
AuthleteApiVersion.parse(configuration.getApiVersion());
// If the specified Authlete API version is "V3".
if (version == AuthleteApiVersion.V3)
{
// An implementation for the Authlete API version 3.
return create(configuration, sKnownImplsV3);
}
else
{
// An implementation for the Authlete API version 2.
return create(configuration, sKnownImplsV2);
}
}
private static AuthleteApi create(AuthleteConfiguration configuration, String[] implementations)
{
for (String className : implementations)
{
try
{
return create(configuration, className);
}
catch (Exception e)
{
// Ignore.
}
}
return null;
}
/**
* Create an instance of {@link AuthleteApi} from the specified class.
*
* @param configuration
* Authlete configuration.
*
* @param className
* The name of a class that implements {@link AuthleteApi}
* interface. The class must have a constructor which takes
* one argument of type {@link AuthleteConfiguration}.
*
* @return
* An instance of the specified class.
*
* @throws IllegalArgumentException
*
* - {@code configuration} is {@code null}.
*
- {@code className} is {@code null}.
*
- The specified class is not found.
*
- The specified class does not implement {@link AuthleteApi} interface.
*
- The specified class does not have a constructor which takes one
* argument of type {@link AuthleteConfiguration}.
*
*
* @throws IllegalStateException
* The constructor of the specified class threw an exception.
*/
public static AuthleteApi create(AuthleteConfiguration configuration, String className)
{
if (configuration == null)
{
throw new IllegalArgumentException("configuration is null.");
}
if (className == null)
{
throw new IllegalArgumentException("className is null.");
}
Class> clazz;
try
{
clazz = Class.forName(className);
}
catch (ClassNotFoundException e)
{
throw new IllegalArgumentException(className + " is not found.", e);
}
if (AuthleteApi.class.isAssignableFrom(clazz) == false)
{
throw new IllegalArgumentException(className + " does not implement AuthleteApi interface.");
}
Constructor> constructor;
try
{
constructor = clazz.getConstructor(AuthleteConfiguration.class);
}
catch (Exception e)
{
throw new IllegalArgumentException(className + " does not have a constructor which takes one AuthleteConfiguration parameter.", e);
}
AuthleteApi api;
try
{
api = (AuthleteApi)constructor.newInstance(configuration);
}
catch (Exception e)
{
throw new IllegalStateException("Failed to create an instance of " + className + ".", e);
}
return api;
}
/**
* Get the default instance of {@link AuthleteApi}.
*
*
* This method loads a configuration file (using {@link
* AuthletePropertiesConfiguration}) on the first call, creates
* an instance of {@link AuthleteApi} and caches the instance.
* The second and subsequent calls return the cached instance.
*
*
*
* The default name of the configuration file is {@code
* authlete.properties}, but it can be changed by a system property
* {@code authlete.configuration.file}. The current directory and
* the classpath are searched for the configuration file in this order.
*
*
* @return
* An instance of {@code AuthleteApi}.
*
* @since 1.29
*/
public static AuthleteApi getDefaultApi()
{
if (sDefaultApi != null)
{
return sDefaultApi;
}
synchronized (AuthleteApiFactory.class)
{
if (sDefaultApi != null)
{
return sDefaultApi;
}
// Load an Authlete configuration file.
AuthleteConfiguration ac = new AuthletePropertiesConfiguration();
// Create an AuthleteApi instance using the configuration.
sDefaultApi = create(ac);
return sDefaultApi;
}
}
}