/**
*
* Copyright (c) 2006-2017, Speedment, Inc. All Rights Reserved.
*
* 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.speedment.runtime.config;
import com.speedment.runtime.config.exception.SpeedmentConfigException;
import com.speedment.runtime.config.mutator.DocumentMutator;
import com.speedment.runtime.config.mutator.ProjectMutator;
import com.speedment.runtime.config.trait.HasChildren;
import com.speedment.runtime.config.trait.HasEnabled;
import com.speedment.runtime.config.trait.HasId;
import com.speedment.runtime.config.trait.HasMainInterface;
import com.speedment.runtime.config.trait.HasMutator;
import com.speedment.runtime.config.trait.HasName;
import com.speedment.runtime.config.trait.HasPackageName;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Optional;
import java.util.regex.Pattern;
import java.util.stream.Stream;
/**
* A typed {@link Document} that represents a database project. A
* {@code Project} is the root of the document tree and can have multiple
* {@link Dbms Dbmses} as children.
*
* @author Emil Forslund
* @since 2.0.0
*/
public interface Project extends
Document,
HasEnabled,
HasId,
HasName,
HasPackageName,
HasChildren,
HasMainInterface,
HasMutator<ProjectMutator<? extends Project>> {
String COMPANY_NAME = "companyName",
PACKAGE_LOCATION = "packageLocation",
CONFIG_PATH = "configPath",
DBMSES = "dbmses",
APP_ID = "appId";
String DEFAULT_COMPANY_NAME = "company",
DEFAULT_PACKAGE_NAME = "com.",
DEFAULT_PACKAGE_LOCATION = "src/main/java/",
DEFAULT_PROJECT_NAME = Project.class.getSimpleName();
/**
* Returns the name of the company that should be used in generated code.
*
* @return the name of the company generating code
*/
default String getCompanyName() {
return getAsString(COMPANY_NAME).orElse(DEFAULT_COMPANY_NAME);
}
/**
* Returns where the code generated for this project will be located.
*
* @return the package location
*/
default String getPackageLocation() {
return getAsString(PACKAGE_LOCATION).orElse(DEFAULT_PACKAGE_LOCATION);
}
/**
* Returns the unique id for this application. This is usually generated on
* application launch. The value should be formatted as a
* {@link java.util.UUID}.
*
* @return the app id
*/
default String getAppId() {
return getAsString(APP_ID).orElseThrow(
() -> new SpeedmentConfigException(
"All applications must have an 'appId' property."
)
);
}
/**
* Returns the path to the configuration file for this project. The
* path may not be set at the time of the calling and the result may
* therefore be {@code empty}.
*
* @return the path to the configuration file
*/
default Optional<Path> getConfigPath() {
return getAsString(CONFIG_PATH).map(Paths::get);
}
/**
* Return a {@code Stream} of all dbmses that exists in this Project.
*
* @return all dbmses
*/
Stream<? extends Dbms> dbmses();
@Override
default Class<Project> mainInterface() {
return Project.class;
}
@Override
default ProjectMutator<? extends Project> mutator() {
return DocumentMutator.of(this);
}
Pattern SPLIT_PATTERN = Pattern.compile("\\."); // Pattern is immutable and therefor thread safe
/**
* Locates the table with the specified full name in this project. The name
* should be separated by dots (.) and have exactly three parts; the name of
* the {@link Dbms}, the name of the {@link Schema} and the name of the
* {@link Table}. If the inputed name is malformed, an
* {@code IllegalArgumentException} will be thrown.
* <p>
* Example of a valid name: {@code db0.socialnetwork.image}
* <p>
* If no table matching the specified name was found, an exception is also
* thrown.
*
* @param fullName the full name of the table
* @return the table found
*/
default Table findTableByName(String fullName) {
final String[] parts = SPLIT_PATTERN.split(fullName);
if (parts.length != 3) {
throw new IllegalArgumentException(
"fullName should consist of three parts separated by dots. "
+ "These are dbms-name, schema-name and table-name."
);
}
final String dbmsId = parts[0],
schemaId = parts[1],
tableId = parts[2];
return dbmses()
.filter(d -> dbmsId.equals(d.getId()))
.findAny()
.orElseThrow(() -> new IllegalArgumentException(
"Could not find dbms: '" + dbmsId + "'."))
.schemas().filter(s -> schemaId.equals(s.getId())).findAny()
.orElseThrow(() -> new IllegalArgumentException(
"Could not find schema: '" + schemaId + "'."))
.tables().filter(t -> tableId.equals(t.getId())).findAny()
.orElseThrow(() -> new IllegalArgumentException(
"Could not find table: '" + tableId + "'."));
}
}