org.neo4j.kernel.impl.newapi.KernelToken Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of neo4j-kernel Show documentation
Show all versions of neo4j-kernel Show documentation
Neo4j kernel is a lightweight, embedded Java database designed to
store data structured as graphs rather than tables. For more
information, see http://neo4j.org.
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [https://neo4j.com]
*
* This file is part of Neo4j.
*
* Neo4j is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package org.neo4j.kernel.impl.newapi;
import static org.neo4j.internal.kernel.api.TokenWrite.checkValidTokenName;
import java.util.function.IntSupplier;
import org.neo4j.exceptions.KernelException;
import org.neo4j.internal.id.IdCapacityExceededException;
import org.neo4j.internal.kernel.api.Token;
import org.neo4j.internal.kernel.api.exceptions.schema.TokenCapacityExceededKernelException;
import org.neo4j.internal.kernel.api.security.AccessMode;
import org.neo4j.internal.kernel.api.security.PrivilegeAction;
import org.neo4j.kernel.api.txstate.TransactionState;
import org.neo4j.kernel.impl.api.KernelTransactionImplementation;
import org.neo4j.storageengine.api.CommandCreationContext;
import org.neo4j.storageengine.api.StorageReader;
import org.neo4j.token.TokenHolders;
import org.neo4j.token.api.NamedToken;
import org.neo4j.token.api.NonUniqueTokenException;
import org.neo4j.token.api.TokenConstants;
import org.neo4j.token.api.TokenHolder;
import org.neo4j.token.api.TokenNotFoundException;
public class KernelToken extends KernelTokenRead implements Token {
private final StorageReader store;
private final CommandCreationContext commandCreationContext;
private final KernelTransactionImplementation ktx;
private final TokenHolders tokenHolders;
public KernelToken(
StorageReader store,
CommandCreationContext commandCreationContext,
KernelTransactionImplementation ktx,
TokenHolders tokenHolders) {
super(store, tokenHolders);
this.store = store;
this.commandCreationContext = commandCreationContext;
this.ktx = ktx;
this.tokenHolders = tokenHolders;
}
@Override
public int labelGetOrCreateForName(String labelName) throws KernelException {
return getOrCreateForName(tokenHolders.labelTokens(), PrivilegeAction.CREATE_LABEL, labelName);
}
@Override
public void labelGetOrCreateForNames(String[] labelNames, int[] labelIds) throws KernelException {
getOrCreateForNames(tokenHolders.labelTokens(), PrivilegeAction.CREATE_LABEL, labelNames, labelIds);
}
@Override
public int labelCreateForName(String labelName, boolean internal) throws KernelException {
ktx.assertOpen();
TransactionState txState = ktx.txState();
int id = reserveTokenId(commandCreationContext::reserveLabelTokenId, tokenHolders.labelTokens());
txState.labelDoCreateForName(labelName, internal, id);
return id;
}
@Override
public int relationshipTypeCreateForName(String relationshipTypeName, boolean internal) throws KernelException {
ktx.assertOpen();
TransactionState txState = ktx.txState();
int id = reserveTokenId(
commandCreationContext::reserveRelationshipTypeTokenId, tokenHolders.relationshipTypeTokens());
txState.relationshipTypeDoCreateForName(relationshipTypeName, internal, id);
return id;
}
@Override
public void relationshipTypeWithSpecificIdCreateForName(int relationshipTypeId, String relationshipTypeName)
throws KernelException {
ktx.assertOpen();
TransactionState txState = ktx.txState();
try {
var existingToken = tokenHolders.relationshipTypeTokens().getTokenById(relationshipTypeId);
var newToken = new NamedToken(relationshipTypeName, relationshipTypeId);
if (!newToken.equals(existingToken)) {
throw new NonUniqueTokenException(TokenHolder.TYPE_RELATIONSHIP_TYPE, newToken, existingToken);
}
} catch (TokenNotFoundException e) {
// There's no relationship type token with the given ID
}
int existingId = tokenHolders.relationshipTypeTokens().getIdByName(relationshipTypeName);
if (existingId != TokenConstants.NO_TOKEN) {
if (relationshipTypeId != existingId) {
throw new NonUniqueTokenException(
TokenHolder.TYPE_RELATIONSHIP_TYPE,
new NamedToken(relationshipTypeName, relationshipTypeId),
new NamedToken(relationshipTypeName, existingId));
}
}
// else there's no relationship type token with the given name
txState.relationshipTypeDoCreateForName(relationshipTypeName, false, relationshipTypeId);
ktx.needsHighIdTracking();
}
@Override
public int propertyKeyCreateForName(String propertyKeyName, boolean internal) throws KernelException {
ktx.assertOpen();
TransactionState txState = ktx.txState();
int id = reserveTokenId(commandCreationContext::reservePropertyKeyTokenId, tokenHolders.propertyKeyTokens());
txState.propertyKeyDoCreateForName(propertyKeyName, internal, id);
return id;
}
@Override
public int propertyKeyGetOrCreateForName(String propertyKeyName) throws KernelException {
return getOrCreateForName(
tokenHolders.propertyKeyTokens(), PrivilegeAction.CREATE_PROPERTYKEY, propertyKeyName);
}
@Override
public void propertyKeyGetOrCreateForNames(String[] propertyKeys, int[] ids) throws KernelException {
getOrCreateForNames(tokenHolders.propertyKeyTokens(), PrivilegeAction.CREATE_PROPERTYKEY, propertyKeys, ids);
}
@Override
public int relationshipTypeGetOrCreateForName(String relationshipTypeName) throws KernelException {
return getOrCreateForName(
tokenHolders.relationshipTypeTokens(), PrivilegeAction.CREATE_RELTYPE, relationshipTypeName);
}
@Override
public void relationshipTypeGetOrCreateForNames(String[] relationshipTypes, int[] ids) throws KernelException {
getOrCreateForNames(
tokenHolders.relationshipTypeTokens(), PrivilegeAction.CREATE_RELTYPE, relationshipTypes, ids);
}
@Override
public int propertyKeyCount() {
ktx.assertOpen();
return store.propertyKeyCount();
}
@Override
public int relationshipTypeCount() {
ktx.assertOpen();
return store.relationshipTypeCount();
}
@Override
void performCheckBeforeOperation() {
ktx.assertOpen();
}
@Override
AccessMode getAccessMode() {
return ktx.securityContext().mode();
}
private int getOrCreateForName(TokenHolder tokens, PrivilegeAction action, String name) throws KernelException {
ktx.assertOpen();
int id = tokens.getIdByName(checkValidTokenName(name));
if (id != NO_TOKEN) {
return id;
}
ktx.securityAuthorizationHandler().assertAllowsTokenCreates(ktx.securityContext(), action);
// ensures the registry has all applied transactions before attempting to create a new one
ktx.ensureValid();
return tokens.getOrCreateId(name);
}
private void getOrCreateForNames(TokenHolder tokenHolder, PrivilegeAction action, String[] names, int[] ids)
throws KernelException {
ktx.assertOpen();
assertSameLength(names, ids);
for (int i = 0; i < names.length; i++) {
ids[i] = tokenHolder.getIdByName(checkValidTokenName(names[i]));
if (ids[i] == NO_TOKEN) {
ktx.securityAuthorizationHandler().assertAllowsTokenCreates(ktx.securityContext(), action);
// ensures the registry has all applied transactions before attempting to create any new ones
ktx.ensureValid();
tokenHolder.getOrCreateIds(names, ids);
return;
}
}
}
private static void assertSameLength(String[] names, int[] ids) {
if (names.length != ids.length) {
throw new IllegalArgumentException("Name and id arrays have different length.");
}
}
private static int reserveTokenId(IntSupplier generator, TokenHolder holder) throws KernelException {
try {
int id;
do {
id = generator.getAsInt();
} while (holder.hasToken(id)); // Retry if id is already taken.
return id;
} catch (IdCapacityExceededException e) {
throw new TokenCapacityExceededKernelException(e, holder.getTokenType());
}
}
}