blob: f35261e16a11d7bc011f25dc22632f4f0fbd24c9 [file] [log] [blame]
/*
* 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.ignite.internal.catalog;
import static it.unimi.dsi.fastutil.ints.Int2ObjectMaps.unmodifiable;
import static java.util.Collections.unmodifiableList;
import static java.util.Comparator.comparingInt;
import static java.util.function.Function.identity;
import static java.util.stream.Collectors.collectingAndThen;
import static java.util.stream.Collectors.toUnmodifiableMap;
import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
import it.unimi.dsi.fastutil.ints.Int2ObjectMap.Entry;
import it.unimi.dsi.fastutil.ints.Int2ObjectMaps;
import it.unimi.dsi.fastutil.ints.Int2ObjectOpenHashMap;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.stream.Collector;
import org.apache.ignite.internal.catalog.descriptors.CatalogIndexDescriptor;
import org.apache.ignite.internal.catalog.descriptors.CatalogObjectDescriptor;
import org.apache.ignite.internal.catalog.descriptors.CatalogSchemaDescriptor;
import org.apache.ignite.internal.catalog.descriptors.CatalogTableDescriptor;
import org.apache.ignite.internal.catalog.descriptors.CatalogZoneDescriptor;
import org.apache.ignite.internal.tostring.IgniteToStringExclude;
import org.apache.ignite.internal.tostring.S;
import org.apache.ignite.internal.util.CollectionUtils;
import org.jetbrains.annotations.Nullable;
/**
* Catalog descriptor represents database schema snapshot.
*/
public class Catalog {
private static <T extends CatalogObjectDescriptor> Collector<T, ?, Map<String, T>> toMapByName() {
return toUnmodifiableMap(CatalogObjectDescriptor::name, identity());
}
private static <T extends CatalogObjectDescriptor> Collector<T, ?, Int2ObjectMap<T>> toMapById() {
return collectingAndThen(
CollectionUtils.toIntMapCollector(CatalogObjectDescriptor::id, identity()),
Int2ObjectMaps::unmodifiable
);
}
private final int version;
private final int objectIdGen;
private final long activationTimestamp;
private final Map<String, CatalogSchemaDescriptor> schemasByName;
private final Map<String, CatalogZoneDescriptor> zonesByName;
private final Map<String, CatalogTableDescriptor> tablesByName;
private final @Nullable CatalogZoneDescriptor defaultZone;
@IgniteToStringExclude
private final Int2ObjectMap<CatalogSchemaDescriptor> schemasById;
@IgniteToStringExclude
private final Int2ObjectMap<CatalogTableDescriptor> tablesById;
@IgniteToStringExclude
private final Int2ObjectMap<CatalogIndexDescriptor> indexesById;
@IgniteToStringExclude
private final Int2ObjectMap<List<CatalogIndexDescriptor>> indexesByTableId;
@IgniteToStringExclude
private final Int2ObjectMap<CatalogZoneDescriptor> zonesById;
/**
* Constructor.
*
* @param version A version of the catalog.
* @param activationTimestamp A timestamp when this version becomes active (i.e. available for use).
* @param objectIdGen Current state of identifier generator. This value should be used to assign an id to a new object in the
* next version of the catalog.
* @param zones Distribution zones descriptors.
* @param schemas Enumeration of schemas available in the current version of catalog.
* @param defaultZoneId ID of the default distribution zone.
*/
public Catalog(
int version,
long activationTimestamp,
int objectIdGen,
Collection<CatalogZoneDescriptor> zones,
Collection<CatalogSchemaDescriptor> schemas,
@Nullable Integer defaultZoneId
) {
this.version = version;
this.activationTimestamp = activationTimestamp;
this.objectIdGen = objectIdGen;
Objects.requireNonNull(schemas, "schemas");
Objects.requireNonNull(zones, "zones");
schemasByName = schemas.stream().collect(toMapByName());
zonesByName = zones.stream().collect(toMapByName());
tablesByName = new HashMap<>();
for (CatalogSchemaDescriptor schema : schemas) {
for (CatalogTableDescriptor table : schema.tables()) {
tablesByName.put(schema.name() + "." + table.name(), table);
}
}
schemasById = schemas.stream().collect(toMapById());
tablesById = schemas.stream().flatMap(s -> Arrays.stream(s.tables())).collect(toMapById());
indexesById = schemas.stream().flatMap(s -> Arrays.stream(s.indexes())).collect(toMapById());
indexesByTableId = unmodifiable(toIndexesByTableId(schemas));
zonesById = zones.stream().collect(toMapById());
if (defaultZoneId != null) {
defaultZone = zonesById.get((int) defaultZoneId);
if (defaultZone == null) {
throw new IllegalStateException("The default zone was not found among the provided zones [id=" + defaultZoneId + ']');
}
} else {
defaultZone = null;
}
}
public int version() {
return version;
}
public long time() {
return activationTimestamp;
}
public int objectIdGenState() {
return objectIdGen;
}
public @Nullable CatalogSchemaDescriptor schema(String name) {
return schemasByName.get(name);
}
public @Nullable CatalogSchemaDescriptor schema(int schemaId) {
return schemasById.get(schemaId);
}
public Collection<CatalogSchemaDescriptor> schemas() {
return schemasByName.values();
}
public @Nullable CatalogTableDescriptor table(int tableId) {
return tablesById.get(tableId);
}
/**
* Returns table descriptor by fully-qualified table name.
*
* @param tableName Fully-qualified table name. Case-sensitive, without quotes.
* */
public @Nullable CatalogTableDescriptor table(String tableName) {
return tablesByName.get(tableName);
}
public Collection<CatalogTableDescriptor> tables() {
return tablesById.values();
}
public @Nullable CatalogIndexDescriptor index(int indexId) {
return indexesById.get(indexId);
}
public Collection<CatalogIndexDescriptor> indexes() {
return indexesById.values();
}
public List<CatalogIndexDescriptor> indexes(int tableId) {
return indexesByTableId.getOrDefault(tableId, List.of());
}
public @Nullable CatalogZoneDescriptor zone(String name) {
return zonesByName.get(name);
}
public @Nullable CatalogZoneDescriptor zone(int zoneId) {
return zonesById.get(zoneId);
}
public Collection<CatalogZoneDescriptor> zones() {
return zonesByName.values();
}
public @Nullable CatalogZoneDescriptor defaultZone() {
return defaultZone;
}
@Override
public String toString() {
return S.toString(this);
}
private static Int2ObjectMap<List<CatalogIndexDescriptor>> toIndexesByTableId(Collection<CatalogSchemaDescriptor> schemas) {
Int2ObjectMap<List<CatalogIndexDescriptor>> indexesByTableId = new Int2ObjectOpenHashMap<>();
for (CatalogSchemaDescriptor schema : schemas) {
for (CatalogIndexDescriptor index : schema.indexes()) {
indexesByTableId.computeIfAbsent(index.tableId(), indexes -> new ArrayList<>()).add(index);
}
}
for (List<CatalogIndexDescriptor> indexes : indexesByTableId.values()) {
indexes.sort(comparingInt(CatalogIndexDescriptor::id));
}
for (Entry<List<CatalogIndexDescriptor>> entry : indexesByTableId.int2ObjectEntrySet()) {
entry.setValue(unmodifiableList(entry.getValue()));
}
return indexesByTableId;
}
}