| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You 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 org.apache.jackrabbit.test; |
| |
| import java.io.File; |
| import java.io.FileInputStream; |
| import java.io.IOException; |
| import java.io.InputStream; |
| import java.lang.reflect.Constructor; |
| import java.security.Principal; |
| import java.util.Map; |
| import java.util.Properties; |
| |
| import javax.jcr.Credentials; |
| import javax.jcr.Repository; |
| import javax.jcr.RepositoryException; |
| import javax.jcr.Session; |
| import javax.jcr.SimpleCredentials; |
| |
| /** |
| * The <code>RepositoryStub</code> is the entry point to the JCR Repository |
| * for the TCK Test harness. |
| * <p> |
| * Implementors of the JCR specification need to provide an implementation |
| * for the abstract methods defined in this class. |
| */ |
| public abstract class RepositoryStub { |
| |
| public static final String STUB_IMPL_PROPS = "repositoryStubImpl.properties"; |
| |
| public static final String PROP_PREFIX = "javax.jcr.tck"; |
| |
| public static final String STUB_IMPL_SYS_PROPS = PROP_PREFIX + ".properties"; |
| |
| public static final String PROP_STUB_IMPL_CLASS = PROP_PREFIX + ".repository_stub_impl"; |
| |
| public static final String PROP_SUPERUSER_PWD = "superuser.pwd"; |
| |
| public static final String PROP_SUPERUSER_NAME = "superuser.name"; |
| |
| public static final String PROP_READONLY_PWD = "readonly.pwd"; |
| |
| public static final String PROP_READONLY_NAME = "readonly.name"; |
| |
| public static final String PROP_READWRITE_PWD = "readwrite.pwd"; |
| |
| public static final String PROP_READWRITE_NAME = "readwrite.name"; |
| |
| public static final String PROP_NODETYPE = "nodetype"; |
| |
| public static final String PROP_NODETYPETESTROOT = "nodetypetestroot"; |
| |
| public static final String PROP_NODETYPENOCHILDREN = "nodetypenochildren"; |
| |
| public static final String PROP_TESTROOT = "testroot"; |
| |
| public static final String PROP_NODE_NAME1 = "nodename1"; |
| |
| public static final String PROP_NODE_NAME2 = "nodename2"; |
| |
| public static final String PROP_NODE_NAME3 = "nodename3"; |
| |
| public static final String PROP_NODE_NAME4 = "nodename4"; |
| |
| public static final String PROP_PROP_NAME1 = "propertyname1"; |
| |
| public static final String PROP_PROP_NAME2 = "propertyname2"; |
| |
| /** |
| * @since JCR 2.0 |
| */ |
| public static final String PROP_PROP_VALUE1 = "propertyvalue1"; |
| |
| /** |
| * @since JCR 2.0 |
| */ |
| public static final String PROP_PROP_VALUE2 = "propertyvalue2"; |
| |
| /** |
| * @since JCR 2.0 |
| */ |
| public static final String PROP_PROP_TYPE1 = "propertytype1"; |
| |
| /** |
| * @since JCR 2.0 |
| */ |
| public static final String PROP_PROP_TYPE2 = "propertytype2"; |
| |
| public static final String PROP_WORKSPACE_NAME = "workspacename"; |
| |
| public static final String PROP_NAMESPACES = "namespaces"; |
| |
| /** |
| * @since JCR 2.0 |
| */ |
| public static final String PROP_LOCK_TIMEOUT = "lock.timeout"; |
| |
| /** |
| * @since JCR 2.0 |
| */ |
| public static final String PROP_LOCK_OWNER = "lock.owner"; |
| |
| /** |
| * Determines whether the repository implementation allows open scoped locks |
| * to be owned by multiple sessions (see JCR 2.0 Section 17.3). |
| * |
| * @since JCR 2.0 |
| */ |
| public static final String PROP_OPEN_SCOPED_LOCK_MULTIPLE = "open.scoped.lock.multiple"; |
| |
| /** |
| * @since JCR 2.0 |
| */ |
| public static final String PROP_HOLD_NAME = "holdname"; |
| |
| public static final String RETENTION_POLICY_HOLDER = "retentionpolicyholder"; |
| |
| /** |
| * @since JCR 2.0 |
| */ |
| public static final String REPOSITORY_FACTORY = "repository.factory"; |
| |
| protected final Properties environment; |
| |
| protected SimpleCredentials superuser; |
| |
| protected SimpleCredentials readonly; |
| |
| protected SimpleCredentials readwrite; |
| |
| /** |
| * Implementations of this class must overwrite this constructor. |
| * |
| * @param env the environment variables. This parameter must not be null. |
| */ |
| protected RepositoryStub(Properties env) { |
| if (env == null) { |
| throw new IllegalArgumentException("Parameter 'env' must not be null!"); |
| } |
| environment = env; |
| superuser = new SimpleCredentials(env.getProperty(PROP_PREFIX + "." + PROP_SUPERUSER_NAME, ""), |
| env.getProperty(PROP_PREFIX + "." + PROP_SUPERUSER_PWD, "").toCharArray()); |
| readonly = new SimpleCredentials(env.getProperty(PROP_PREFIX + "." + PROP_READONLY_NAME, ""), |
| env.getProperty(PROP_PREFIX + "." + PROP_READONLY_PWD, "").toCharArray()); |
| readwrite = new SimpleCredentials(env.getProperty(PROP_PREFIX + "." + PROP_READWRITE_NAME, ""), |
| env.getProperty(PROP_PREFIX + "." + PROP_READWRITE_PWD, "").toCharArray()); |
| } |
| |
| /** |
| * Creates and/or returns the configured <code>RepositryStub</code> |
| * implementation. |
| * <p> |
| * The property file is located in the following sequence: |
| * <ol> |
| * <li>If the system property <code>-Djavax.jcr.tck.properties</code> is |
| * set, then the accroding file is used as configuration.</li> |
| * <li>If the system property <code>-Djavax.jcr.tck.properties</code> is |
| * not set, then the TCK tries to load the file <code>repositoryStubImpl.properties</code> |
| * as a resource from the ClassLoader of this <code>RepositryStub</code> class.</li> |
| * </ol> |
| * The properties are then overlayed with the the key / values from |
| * <code>configuration</code> map. |
| * |
| * @param configuration a <code>Map</code> of additional configuration entries. |
| * @return a <code>RepositoryStub</code> implementation. |
| * @throws RepositoryStubException |
| */ |
| public static synchronized RepositoryStub getInstance(Map configuration) |
| throws RepositoryStubException { |
| Properties props = null; |
| RepositoryStub stub = null; |
| String implProp = System.getProperty(STUB_IMPL_SYS_PROPS); |
| if (implProp != null) { |
| File implPropFile = new File(implProp); |
| if (implPropFile.exists()) { |
| props = new Properties(); |
| try { |
| props.load(new FileInputStream(implPropFile)); |
| } catch (IOException e) { |
| throw new RepositoryStubException( |
| "Unable to load config file: " + implProp, e); |
| } |
| } else { |
| throw new RepositoryStubException("File does not exist: " + implProp); |
| } |
| } |
| |
| if (props == null) { |
| props = new Properties(); |
| InputStream is = RepositoryStub.class.getClassLoader().getResourceAsStream(STUB_IMPL_PROPS); |
| if (is != null) { |
| try { |
| props.load(is); |
| } catch (IOException e) { |
| throw new RepositoryStubException( |
| "Exception reading " + STUB_IMPL_PROPS, e); |
| } |
| } |
| } |
| |
| // overlay with configuration parameter |
| props.putAll(configuration); |
| |
| try { |
| String className = props.getProperty(PROP_STUB_IMPL_CLASS); |
| if (className == null || className.length() == 0) { |
| throw new RepositoryStubException("Property " + PROP_STUB_IMPL_CLASS + " not defined!"); |
| } |
| Class stubClass = Class.forName(className); |
| Constructor constr = stubClass.getConstructor(new Class[]{Properties.class}); |
| stub = (RepositoryStub) constr.newInstance(new Object[]{props}); |
| } catch (Exception e) { |
| throw new RepositoryStubException(e); |
| } |
| |
| return stub; |
| } |
| |
| /** |
| * Returns a reference to the <code>Repository</code> provided by this |
| * <code>RepositoryStub</code>. |
| * |
| * @return |
| */ |
| public abstract Repository getRepository() throws RepositoryStubException; |
| |
| /** |
| * Returns a <code>Credentials</code> object, that can be used to login |
| * to the <code>Repository</code> returned by {@link #getRepository}. |
| * <p> |
| * The <code>Credentials</code> returned has 'superuser' rights. That |
| * is, the <code>Session</code> object returned by {@link Repository#login(Credentials)} |
| * has read write access to the whole Content Repository. |
| * |
| * @return a <code>Credentials</code> object, that allows to login to the |
| * <code>Repository</code> as 'superuser'. |
| */ |
| public Credentials getSuperuserCredentials() { |
| return superuser; |
| } |
| |
| /** |
| * Returns a <code>Credentials</code> object, that can be used to login |
| * to the <code>Repository</code> returned by {@link #getRepository}. |
| * <p> |
| * The <code>Credentials</code> returned has read/write rights. That |
| * is, the <code>Session</code> object returned by {@link Repository#login(Credentials)} |
| * has read write access to the <code>Node</code> configured in the |
| * JCR TCK Interview. |
| * <p> |
| * For details, see: JCR TCK User Guide. |
| * |
| * @return a <code>Credentials</code> object, that allows to login to the |
| * <code>Repository</code> with read/write right. |
| */ |
| public Credentials getReadWriteCredentials() { |
| return readwrite; |
| } |
| |
| /** |
| * Returns a <code>Credentials</code> object, that can be used to login |
| * to the <code>Repository</code> returned by {@link #getRepository}. |
| * <p> |
| * The <code>Credentials</code> returned must have read-only rights. That |
| * is, the <code>Session</code> object returned by {@link Repository#login()} |
| * has read-only access to the <code>Node</code> configured in the |
| * JCR TCK Interview. |
| * <p> |
| * For details, see: JCR TCK User Guide. |
| * |
| * @return a <code>Credentials</code> object, that allows to login to the |
| * <code>Repository</code> with read-only right. |
| */ |
| public Credentials getReadOnlyCredentials() { |
| return readonly; |
| } |
| |
| /** |
| * Returns the property with the specified <code>name</code>. If a |
| * property with the given name does not exist, <code>null</code> is |
| * returned. |
| * @param name the name of the property. |
| * @return the property, or <code>null</code> if the property does not |
| * exist. |
| */ |
| public String getProperty(String name) { |
| return environment.getProperty(name); |
| } |
| |
| /** |
| * Returns a {@link Principal} identifiying a known user (not group) |
| * @param session |
| */ |
| public abstract Principal getKnownPrincipal(Session session) throws RepositoryException; |
| |
| /** |
| * Returns a {@link Principal} identifiying an unknown user. |
| * @param session |
| * @throws NotExecutableException if the implementation can not provide an |
| * instance of Principal for unkwnown users |
| */ |
| public abstract Principal getUnknownPrincipal(Session session) throws RepositoryException, NotExecutableException; |
| } |