All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.teamapps.universaldb.schema.Schema Maven / Gradle / Ivy

/*-
 * ========================LICENSE_START=================================
 * UniversalDB
 * ---
 * Copyright (C) 2014 - 2024 TeamApps.org
 * ---
 * 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.
 * =========================LICENSE_END==================================
 */
package org.teamapps.universaldb.schema;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.teamapps.universaldb.TableConfig;
import org.teamapps.universaldb.index.ColumnType;
import org.teamapps.universaldb.util.DataStreamUtil;

import java.io.ByteArrayInputStream;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.lang.invoke.MethodHandles;
import java.nio.charset.StandardCharsets;
import java.util.*;
import java.util.stream.Collectors;

public class Schema {
	private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());

	private final int schemaVersion = 1;
	private String pojoNamespace = "org.teamapps.datamodel";
	private String schemaName = "SchemaInfo";
	private final List databases = new ArrayList<>();

	public static Schema create() {
		return new Schema();
	}

	public static Schema create(String pojoNamespace) {
		Schema schema = new Schema();
		schema.setPojoNamespace(pojoNamespace);
		return schema;
	}

	public static Schema parse(String schemaData) {
		return new Schema(schemaData);
	}

	public static void checkName(String name) {
		if (name == null || name.isEmpty() || !name.chars().allMatch(Character::isLetterOrDigit)) {
			throw new RuntimeException("ERROR: INVALID NAME:" + name);
		}
	}

	public static void checkColumnName(String name) {
		checkName(name);
		if (Table.isReservedMetaName(name)) {
			throw new RuntimeException("ERROR: FORBIDDEN COLUMN NAME:" + name);
		}
	}

	private static List getTokens(String line) {
		line = line.trim();
		String[] parts = line.split(" ");
		return Arrays.asList(parts);
	}

	public Schema() {
	}

	private Schema(String schemaData) {
		String[] lines = schemaData.split("\n");
		boolean foundSchema = false;
		Database db = null;
		Table table = null;
		Set columnTypes = ColumnType.getNames();
		Map unresolvedReferenceTableMap = new HashMap<>();
		for (String line : lines) {
			int mappingId = 0;
			line = line.trim();
			if (line.startsWith("/") || line.startsWith("#")) {
				continue;
			}
			if (line.endsWith("]")) {
				int start = line.lastIndexOf('[');
				int end = line.lastIndexOf(']');
				mappingId = Integer.parseInt(line.substring(start + 1, end));
				line = line.substring(0, start);
			}
			List tokens = getTokens(line);
			if (tokens.size() < 3 || !tokens.get(1).equalsIgnoreCase("as")) {
				continue;
			}
			String name = tokens.get(0);
			String type = tokens.get(2);
			if (type.equalsIgnoreCase("SCHEMA")) {
				foundSchema = true;
				setPojoNamespace(name);
				if (tokens.size() > 3) {
					String schemaName = tokens.get(3);
					if (!schemaName.isBlank()) {
						setSchemaName(schemaName);
					}
				}
			}
			if (!foundSchema) {
				continue;
			}
			if (type.equalsIgnoreCase("DATABASE")) {
				db = addDatabase(name);
				db.setMappingId(mappingId);
			}
			if (type.equalsIgnoreCase("TABLE")) {
				TableConfig tableConfig = TableConfig.parse(line);
				table = db.addTable(name, tableConfig.getTableOptions());
				table.setMappingId(mappingId);
			}
			if (type.equalsIgnoreCase("VIEW")) {
				String referencedTablePath = tokens.get(4);
				table = db.addView(name, referencedTablePath);
				table.setMappingId(mappingId);
			}
			if (type.equalsIgnoreCase("ENUM")) {
				//enable separate enum definition...
			}
			if (columnTypes.contains(type.toUpperCase()) && !Table.isReservedMetaName(name)) {
				ColumnType columnType = ColumnType.valueOf(type);
				if (table != null) {
					Column column = table.addColumn(name, columnType);
					column.setMappingId(mappingId);
					if (columnType.isReference()) {
						unresolvedReferenceTableMap.put(column, tokens.get(3));
						String backReference = tokens.get(5);
						column.setBackReference(backReference.equals("NONE") ? null : backReference);
						if (line.contains("CASCADE DELETE REFERENCES")) {
							column.setCascadeDeleteReferences(true);
						}
					} else if (columnType == ColumnType.ENUM) {
						line = line.trim();
						String[] values = line.substring(line.indexOf("VALUES (") + 8).replace(")", "").split(", ");
						column.setEnumValues(Arrays.asList(values));
					}
				}
			} else if (columnTypes.contains(type.toUpperCase()) && Table.isReservedMetaName(name)) {
				if (table != null) {
					Column column = table.getColumn(name);
					column.setMappingId(mappingId);
				}
			}
		}
		for (Map.Entry entry : unresolvedReferenceTableMap.entrySet()) {
			Column column = entry.getKey();
			String fullName = entry.getValue();
			String[] parts = fullName.split("\\.");
			String dbName = parts[0];
			String tableName = parts[1];
			Database refDb = getDatabases().stream().filter(database -> database.getName().equals(dbName)).findAny().orElse(null);
			Table referenceTable = refDb.getAllTables().stream().filter(refTable -> refTable.getName().equals(tableName)).findAny().orElse(null);
			column.setReferencedTable(referenceTable);
		}
	}

	public Schema(byte[] data) throws IOException {
		this(new DataInputStream(new ByteArrayInputStream(data)));
	}

	public Schema(DataInputStream dis) throws IOException {
		this(DataStreamUtil.readStringWithLengthHeader(dis));
	}

	public byte[] getSchemaData() {
		String schema = getSchemaDefinition();
		return schema.getBytes(StandardCharsets.UTF_8);
	}

	public void writeSchema(DataOutputStream dataOutputStream) throws IOException {
		byte[] schemaData = getSchemaData();
		DataStreamUtil.writeByteArrayWithLengthHeader(dataOutputStream, schemaData);
	}

	private String createDefinition(boolean ignoreMapping) {
		StringBuilder sb = new StringBuilder();
		sb.append(pojoNamespace).append(" as SCHEMA").append("\n");
		databases.forEach(db -> sb.append(db.createDefinition(ignoreMapping)));
		return sb.toString();
	}

	public List getDatabases() {
		return databases;
	}

	public Database getDatabase(String name) {
		return databases.stream()
				.filter(db -> db.getName().equals(name))
				.findFirst()
				.orElse(null);
	}

	public Database addDatabase(String name) {
		checkName(name);
		return addDatabase(new Database(this, name));
	}

	private Database addDatabase(Database dataBase) {
		databases.add(dataBase);
		return dataBase;
	}

	public String getPojoNamespace() {
		return pojoNamespace;
	}

	public void setPojoNamespace(String pojoNamespace) {
		this.pojoNamespace = pojoNamespace;
	}

	public String getSchemaName() {
		return schemaName;
	}

	public void setSchemaName(String schemaName) {
		this.schemaName = schemaName;
	}

	public int getSchemaVersion() {
		return schemaVersion;
	}

	public boolean isCompatibleWith(Schema schema) {
		if (this.schemaVersion != schema.getSchemaVersion()) {
			return false;
		}
		Map databaseMap = databases.stream().collect(Collectors.toMap(Database::getName, db -> db));
		for (Database database : schema.getDatabases()) {
			Database localDatabase = databaseMap.get(database.getName());
			if (localDatabase != null) {
				if (localDatabase.getMappingId() > 0 && database.getMappingId() > 0 && localDatabase.getMappingId() != database.getMappingId()) {
					return false;
				}
				boolean compatibleWith = localDatabase.isCompatibleWith(database);
				if (!compatibleWith) {
					return false;
				}
			} else {
				if (database.getMappingId() != 0 && getMappingIds().contains(database.getMappingId())) {
					return false;
				}
			}
		}
		return true;
	}

	public boolean isSameSchema(Schema schema) throws IOException {
		byte[] schemaData = getSchemaData();
		byte[] schemaData2 = schema.getSchemaData();
		return Arrays.equals(schemaData, schemaData2);
	}

	public boolean isSameSchemaIgnoreMapping(Schema schema) {
		return createDefinition(true).equals(schema.createDefinition(true));
	}

	public void merge(Schema schema) {
		if (!isCompatibleWith(schema)) {
			throw new RuntimeException("Error: cannot merge incompatible schemas:" + this + " with " + schema);
		}
		Map databaseMap = databases.stream().collect(Collectors.toMap(Database::getName, db -> db));
		for (Database database : schema.getDatabases()) {
			Database localDatabase = databaseMap.get(database.getName());
			if (localDatabase == null) {
				addDatabase(database);
			} else {
				if (localDatabase.getMappingId() == 0) {
					localDatabase.setMappingId(database.getMappingId());
				}
				localDatabase.merge(database);
			}
		}
	}

	public boolean checkModel() {
		Set mappingIds = new HashSet<>();
		for (Database database : databases) {
			if (database.getMappingId() == 0) {
				logger.error("Missing mapping id:" + database.getFQN());
				return false;
			} else if (mappingIds.contains(database.getMappingId())) {
				logger.error("Duplicate mapping id:" + database.getFQN());
				return false;
			}
			mappingIds.add(database.getMappingId());
			for (Table table : database.getTables()) {
				if (table.getMappingId() == 0) {
					logger.error("Missing mapping id:" + table.getFQN());
					return false;
				} else if (mappingIds.contains(table.getMappingId())) {
					logger.error("Duplicate mapping id:" + table.getFQN());
					return false;
				}
				mappingIds.add(table.getMappingId());
				for (Column columnIndex : table.getColumns()) {
					if (columnIndex.getMappingId() == 0) {
						logger.error("Missing mapping id:" + columnIndex.getFQN());
						return false;
					} else if (mappingIds.contains(columnIndex.getMappingId())) {
						logger.error("Duplicate mapping id:" + columnIndex.getFQN());
						return false;
					}
					mappingIds.add(columnIndex.getMappingId());
				}
			}
		}
		return true;
	}

	public void mapSchema() {
		for (Database database : databases) {
			if (database.getMappingId() == 0) {
				database.setMappingId(getNextMappingId());
			}
			for (Table table : database.getTables()) {
				if (table.getMappingId() == 0) {
					table.setMappingId(getNextMappingId());
				}
				for (Column column : table.getColumns()) {
					if (column.getMappingId() == 0) {
						column.setMappingId(getNextMappingId());
					}
				}
			}
		}
	}

	private int getNextMappingId() {
		Set mappingIds = getMappingIds();
		int mappingId = mappingIds.size() + 1;
		while (mappingIds.contains(mappingId)) {
			mappingId++;
		}
		return mappingId;
	}

	public Set getMappingIds() {
		Set mappingIds = new HashSet<>();
		for (Database database : databases) {
			mappingIds.add(database.getMappingId());
			for (Table table : database.getTables()) {
				mappingIds.add(table.getMappingId());
				for (Column column : table.getColumns()) {
					mappingIds.add(column.getMappingId());
				}
			}
		}
		mappingIds.remove(0);
		return mappingIds;
	}

	@Override
	public String toString() {
		return createDefinition(false);
	}


	public String getSchemaDefinition() {
		return createDefinition(false);
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy