com.arcadedb.database.TransactionContext Maven / Gradle / Ivy
/*
* Copyright © 2021-present Arcade Data Ltd ([email protected])
*
* 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.
*
* SPDX-FileCopyrightText: 2021-present Arcade Data Ltd ([email protected])
* SPDX-License-Identifier: Apache-2.0
*/
package com.arcadedb.database;
import com.arcadedb.GlobalConfiguration;
import com.arcadedb.engine.BasePage;
import com.arcadedb.engine.Bucket;
import com.arcadedb.engine.ImmutablePage;
import com.arcadedb.engine.MutablePage;
import com.arcadedb.engine.PageId;
import com.arcadedb.engine.PageManager;
import com.arcadedb.engine.PaginatedComponent;
import com.arcadedb.engine.PaginatedFile;
import com.arcadedb.engine.WALFile;
import com.arcadedb.exception.ConcurrentModificationException;
import com.arcadedb.exception.DuplicatedKeyException;
import com.arcadedb.exception.RecordNotFoundException;
import com.arcadedb.exception.SchemaException;
import com.arcadedb.exception.TransactionException;
import com.arcadedb.index.IndexInternal;
import com.arcadedb.index.lsm.LSMTreeIndexAbstract;
import com.arcadedb.log.LogManager;
import java.io.*;
import java.util.*;
import java.util.logging.*;
/**
* Manage the transaction context. When the transaction begins, the modifiedPages map is initialized. This allows to always delegate
* to the transaction context, even if there is no active transaction by ignoring tx data.
*
* At commit time, the files are locked in order (to avoid deadlocks) and to allow parallel commit on different files.
*
* Format of WAL:
*
* txId:long|pages:int|<segmentSize:int|fileId:int|pageNumber:long|pageModifiedFrom:int|pageModifiedTo:int|<prevContent><newContent>segmentSize:int>MagicNumber:long
*/
public class TransactionContext implements Transaction {
private final DatabaseInternal database;
private final Map newPageCounters = new HashMap<>();
private final Map immutableRecordsCache = new HashMap<>(1024);
private final Map modifiedRecordsCache = new HashMap<>(1024);
private final TransactionIndexContext indexChanges;
private final Map immutablePages = new HashMap<>(64);
private Map modifiedPages;
private Map newPages;
private boolean useWAL;
private boolean asyncFlush = true;
private WALFile.FLUSH_TYPE walFlush;
private List lockedFiles;
private long txId = -1;
private STATUS status = STATUS.INACTIVE;
// KEEPS TRACK OF MODIFIED RECORD IN TX. AT 1ST PHASE COMMIT TIME THE RECORD ARE SERIALIZED AND INDEXES UPDATED. THIS DEFERRING IMPROVES SPEED ESPECIALLY
// WITH GRAPHS WHERE EDGES ARE CREATED AND CHUNKS ARE UPDATED MULTIPLE TIMES IN THE SAME TX
// TODO: OPTIMIZE modifiedRecordsCache STRUCTURE, MAYBE JOIN IT WITH UPDATED RECORDS?
private Map updatedRecords = null;
private Database.TRANSACTION_ISOLATION_LEVEL isolationLevel = Database.TRANSACTION_ISOLATION_LEVEL.READ_COMMITTED;
public enum STATUS {INACTIVE, BEGUN, COMMIT_1ST_PHASE, COMMIT_2ND_PHASE}
public static class TransactionPhase1 {
public final Binary result;
public final List modifiedPages;
public TransactionPhase1(final Binary result, final List modifiedPages) {
this.result = result;
this.modifiedPages = modifiedPages;
}
}
public TransactionContext(final DatabaseInternal database) {
this.database = database;
this.walFlush = WALFile.getWALFlushType(database.getConfiguration().getValueAsInteger(GlobalConfiguration.TX_WAL_FLUSH));
this.useWAL = database.getConfiguration().getValueAsBoolean(GlobalConfiguration.TX_WAL);
this.indexChanges = new TransactionIndexContext(database);
}
@Override
public void begin(final Database.TRANSACTION_ISOLATION_LEVEL isolationLevel) {
this.isolationLevel = isolationLevel;
if (status != STATUS.INACTIVE)
throw new TransactionException("Transaction already begun");
status = STATUS.BEGUN;
modifiedPages = new HashMap<>();
if (newPages == null)
// KEEP ORDERING IN CASE MULTIPLE PAGES FOR THE SAME FILE ARE CREATED
newPages = new LinkedHashMap<>();
}
@Override
public Binary commit() {
if (status == STATUS.INACTIVE)
throw new TransactionException("Transaction not begun");
if (status != STATUS.BEGUN)
throw new TransactionException("Transaction already in commit phase");
final TransactionPhase1 phase1 = commit1stPhase(true);
if (phase1 != null)
commit2ndPhase(phase1);
else
reset();
if (database.getSchema().getEmbedded().isDirty())
database.getSchema().getEmbedded().saveConfiguration();
return phase1 != null ? phase1.result : null;
}
public Record getRecordFromCache(final RID rid) {
Record rec = modifiedRecordsCache.get(rid);
if (rec == null)
rec = immutableRecordsCache.get(rid);
if (rec == null && updatedRecords != null)
// IN CASE `READ-YOUR-WRITE` IS FALSE, THE MODIFIED RECORD IS NOT IN CACHE AND MUST BE READ FROM THE UPDATE RECORDS
rec = updatedRecords.get(rid);
return rec;
}
public void updateRecordInCache(final Record record) {
if (database.isReadYourWrites()) {
final RID rid = record.getIdentity();
if (rid == null)
throw new IllegalArgumentException("Cannot update record in TX cache because it is not persistent: " + record);
if (record instanceof RecordInternal)
modifiedRecordsCache.put(rid, record);
else
immutableRecordsCache.put(rid, record);
}
}
public void removeImmutableRecordsOfSamePage(final RID rid) {
final int bucketId = rid.getBucketId();
final long pos = rid.getPosition();
final Bucket bucket = database.getSchema().getBucketById(bucketId);
final long pageNum = pos / bucket.getMaxRecordsInPage();
// FOR IMMUTABLE RECORDS AVOID THAT THEY ARE POINTING TO THE OLD OFFSET IN A MODIFIED PAGE
immutableRecordsCache.values()
.removeIf(r -> r.getIdentity().getBucketId() == bucketId && r.getIdentity().getPosition() / bucket.getMaxRecordsInPage() == pageNum);
}
public void removeRecordFromCache(final RID rid) {
if (updatedRecords != null)
updatedRecords.remove(rid);
if (database.isReadYourWrites()) {
if (rid == null)
throw new IllegalArgumentException("Cannot remove record in TX cache because it is not persistent");
modifiedRecordsCache.remove(rid);
immutableRecordsCache.remove(rid);
}
removeImmutableRecordsOfSamePage(rid);
}
public DatabaseInternal getDatabase() {
return database;
}
@Override
public void setUseWAL(final boolean useWAL) {
this.useWAL = useWAL;
}
@Override
public void setWALFlush(final WALFile.FLUSH_TYPE flush) {
this.walFlush = flush;
}
@Override
public void rollback() {
LogManager.instance()
.log(this, Level.FINE, "Rollback transaction newPages=%s modifiedPages=%s (threadId=%d)", newPages, modifiedPages, Thread.currentThread().getId());
if (database.isOpen() && database.getSchema().getDictionary() != null) {
if (modifiedPages != null) {
final int dictionaryId = database.getSchema().getDictionary().getId();
for (final PageId pageId : modifiedPages.keySet()) {
if (dictionaryId == pageId.getFileId()) {
// RELOAD THE DICTIONARY
try {
database.getSchema().getDictionary().reload();
} catch (final IOException e) {
throw new SchemaException("Error on reloading schema dictionary");
}
break;
}
}
}
}
modifiedPages = null;
newPages = null;
updatedRecords = null;
// RELOAD PREVIOUS VERSION OF MODIFIED RECORDS
if (database.isOpen())
for (final Record r : modifiedRecordsCache.values())
try {
r.reload();
} catch (final Exception e) {
// IGNORE EXCEPTION (RECORD DELETED OR TYPE REMOVED)
}
reset();
}
public void assureIsActive() {
if (!isActive())
throw new TransactionException("Transaction not begun");
}
public void addUpdatedRecord(final Record record) throws IOException {
final RID rid = record.getIdentity();
if (updatedRecords == null)
updatedRecords = new HashMap<>();
if (updatedRecords.put(record.getIdentity(), record) == null)
database.getSchema().getBucketById(rid.getBucketId()).fetchPageInTransaction(rid);
updateRecordInCache(record);
removeImmutableRecordsOfSamePage(record.getIdentity());
}
/**
* Looks for the page in the TX context first, then delegates to the database.
*/
public BasePage getPage(final PageId pageId, final int size) throws IOException {
BasePage page = null;
if (modifiedPages != null)
page = modifiedPages.get(pageId);
if (page == null && newPages != null)
page = newPages.get(pageId);
if (page == null)
page = immutablePages.get(pageId);
if (page == null) {
// NOT FOUND, DELEGATES TO THE DATABASE
page = database.getPageManager().getImmutablePage(pageId, size, false, true);
if (page != null) {
switch (isolationLevel) {
case READ_COMMITTED:
break;
case REPEATABLE_READ:
final PaginatedFile file = database.getFileManager().getFile(pageId.getFileId());
final boolean isNewPage = pageId.getPageNumber() >= file.getTotalPages();
if (!isNewPage)
// CACHE THE IMMUTABLE PAGE ONLY IF IT IS NOT NEW
immutablePages.put(pageId, (ImmutablePage) page);
break;
}
}
}
return page;
}
/**
* If the page is not already in transaction tx, loads from the database and clone it locally.
*/
public MutablePage getPageToModify(final PageId pageId, final int size, final boolean isNew) throws IOException {
if (!isActive())
throw new TransactionException("Transaction not active");
MutablePage page = modifiedPages != null ? modifiedPages.get(pageId) : null;
if (page == null) {
if (newPages != null)
page = newPages.get(pageId);
if (page == null) {
// IF AVAILABLE REMOVE THE PAGE FROM IMMUTABLE PAGES TO KEEP ONLY ONE PAGE IN RAM
final ImmutablePage loadedPage = immutablePages.remove(pageId);
if (loadedPage == null)
// NOT FOUND, DELEGATES TO THE DATABASE
page = database.getPageManager().getMutablePage(pageId, size, isNew, true);
else
page = loadedPage.modify();
if (isNew)
newPages.put(pageId, page);
else
modifiedPages.put(pageId, page);
}
}
return page;
}
/**
* Puts the page in the TX modified pages.
*/
public MutablePage getPageToModify(final BasePage page) throws IOException {
if (!isActive())
throw new TransactionException("Transaction not active");
final MutablePage mutablePage = page.modify();
final PageId pageId = page.getPageId();
if (newPages.containsKey(pageId))
newPages.put(pageId, mutablePage);
else
modifiedPages.put(pageId, mutablePage);
immutablePages.remove(pageId);
return mutablePage;
}
public MutablePage addPage(final PageId pageId, final int pageSize) {
assureIsActive();
// CREATE A PAGE ID BASED ON NEW PAGES IN TX. IN CASE OF ROLLBACK THEY ARE SIMPLY REMOVED AND THE GLOBAL PAGE COUNT IS UNCHANGED
final MutablePage page = new MutablePage(database.getPageManager(), pageId, pageSize);
newPages.put(pageId, page);
final Integer indexCounter = newPageCounters.get(pageId.getFileId());
if (indexCounter == null || indexCounter < pageId.getPageNumber() + 1)
newPageCounters.put(pageId.getFileId(), pageId.getPageNumber() + 1);
return page;
}
public long getFileSize(final int fileId) throws IOException {
final Integer lastPage = newPageCounters.get(fileId);
if (lastPage != null)
return (long) (lastPage + 1) * database.getFileManager().getFile(fileId).getPageSize();
return database.getFileManager().getVirtualFileSize(fileId);
}
public Integer getPageCounter(final int indexFileId) {
return newPageCounters.get(indexFileId);
}
@Override
public boolean isActive() {
return status != STATUS.INACTIVE;
}
public Map getStats() {
final LinkedHashMap map = new LinkedHashMap<>();
final Set involvedFiles = new LinkedHashSet<>();
if (modifiedPages != null)
for (final PageId pid : modifiedPages.keySet())
involvedFiles.add(pid.getFileId());
if (newPages != null)
for (final PageId pid : newPages.keySet())
involvedFiles.add(pid.getFileId());
involvedFiles.addAll(newPageCounters.keySet());
map.put("status", status.name());
map.put("involvedFiles", involvedFiles);
map.put("modifiedPages", modifiedPages != null ? modifiedPages.size() : 0);
map.put("newPages", newPages != null ? newPages.size() : 0);
map.put("updatedRecords", updatedRecords != null ? updatedRecords.size() : 0);
map.put("newPageCounters", newPageCounters);
map.put("indexChanges", indexChanges != null ? indexChanges.getTotalEntries() : 0);
return map;
}
public int getModifiedPages() {
int result = 0;
if (modifiedPages != null)
result += modifiedPages.size();
if (newPages != null)
result += newPages.size();
return result;
}
/**
* Test only API.
*/
public void kill() {
lockedFiles = null;
modifiedPages = null;
newPages = null;
updatedRecords = null;
newPageCounters.clear();
immutablePages.clear();
}
/**
* Executes 1st phase from a replica.
*/
public void commitFromReplica(final WALFile.WALTransaction buffer,
final Map>> keysTx)
throws TransactionException {
final int totalImpactedPages = buffer.pages.length;
if (totalImpactedPages == 0 && keysTx.isEmpty()) {
// EMPTY TRANSACTION = NO CHANGES
modifiedPages = null;
return;
}
try {
// LOCK FILES (IN ORDER, SO TO AVOID DEADLOCK)
final Set modifiedFiles = new HashSet<>();
for (final WALFile.WALPage p : buffer.pages)
modifiedFiles.add(p.fileId);
indexChanges.setKeys(keysTx);
indexChanges.addFilesToLock(modifiedFiles);
final int dictionaryFileId = database.getSchema().getDictionary().getId();
boolean dictionaryModified = false;
for (final WALFile.WALPage p : buffer.pages) {
final PaginatedFile file = database.getFileManager().getFile(p.fileId);
final int pageSize = file.getPageSize();
final PageId pageId = new PageId(p.fileId, p.pageNumber);
final boolean isNew = p.pageNumber >= file.getTotalPages();
final MutablePage page = getPageToModify(pageId, pageSize, isNew);
// APPLY THE CHANGE TO THE PAGE
page.writeByteArray(p.changesFrom - BasePage.PAGE_HEADER_SIZE, p.currentContent.content);
page.setContentSize(p.currentPageSize);
if (isNew) {
newPages.put(pageId, page);
newPageCounters.put(pageId.getFileId(), pageId.getPageNumber() + 1);
} else
modifiedPages.put(pageId, page);
if (!dictionaryModified && dictionaryFileId == pageId.getFileId())
dictionaryModified = true;
}
database.commit();
if (dictionaryModified)
database.getSchema().getDictionary().reload();
} catch (final ConcurrentModificationException e) {
rollback();
throw e;
} catch (final Exception e) {
rollback();
throw new TransactionException("Transaction error on commit", e);
}
}
/**
* Locks the files in order, then checks all the pre-conditions.
*/
public TransactionPhase1 commit1stPhase(final boolean isLeader) {
if (status == STATUS.INACTIVE)
throw new TransactionException("Transaction not started");
if (status != STATUS.BEGUN)
throw new TransactionException("Transaction in phase " + status);
if (updatedRecords != null) {
for (final Record rec : updatedRecords.values())
try {
database.updateRecordNoLock(rec, false);
} catch (final RecordNotFoundException e) {
// DELETED IN TRANSACTION, THIS IS FULLY MANAGED TO NEVER HAPPEN, BUT IF IT DOES DUE TO THE INTRODUCTION OF A BUG, JUST LOG SOMETHING AND MOVE ON
LogManager.instance().log(this, Level.WARNING, "Attempt to update the delete record %s in transaction", rec.getIdentity());
}
updatedRecords = null;
}
final int totalImpactedPages = modifiedPages.size() + (newPages != null ? newPages.size() : 0);
if (totalImpactedPages == 0 && indexChanges.isEmpty()) {
// EMPTY TRANSACTION = NO CHANGES
return null;
}
status = STATUS.COMMIT_1ST_PHASE;
if (isLeader)
// LOCK FILES IN ORDER (TO AVOID DEADLOCK)
lockedFiles = lockFilesInOrder();
else
// IN CASE OF REPLICA THIS IS DEMANDED TO THE LEADER EXECUTION
lockedFiles = new ArrayList<>();
try {
if (isLeader)
// COMMIT INDEX CHANGES (IN CASE OF REPLICA THIS IS DEMANDED TO THE LEADER EXECUTION)
indexChanges.commit();
// CHECK THE VERSIONS FIRST
final List pages = new ArrayList<>();
final PageManager pageManager = database.getPageManager();
for (final Iterator it = modifiedPages.values().iterator(); it.hasNext(); ) {
final MutablePage p = it.next();
final int[] range = p.getModifiedRange();
if (range[1] > 0) {
pageManager.checkPageVersion(p, false);
pages.add(p);
} else
// PAGE NOT MODIFIED, REMOVE IT
it.remove();
}
if (newPages != null)
for (final MutablePage p : newPages.values()) {
final int[] range = p.getModifiedRange();
if (range[1] > 0) {
pageManager.checkPageVersion(p, true);
pages.add(p);
}
}
Binary result = null;
if (useWAL) {
txId = database.getTransactionManager().getNextTransactionId();
//LogManager.instance().log(this, Level.FINE, "Creating buffer for TX %d (threadId=%d)", txId, Thread.currentThread().getId());
result = database.getTransactionManager().createTransactionBuffer(txId, pages);
}
return new TransactionPhase1(result, pages);
} catch (final DuplicatedKeyException | ConcurrentModificationException e) {
rollback();
throw e;
} catch (final Exception e) {
LogManager.instance().log(this, Level.FINE, "Unknown exception during commit (threadId=%d)", e, Thread.currentThread().getId());
rollback();
throw new TransactionException("Transaction error on commit", e);
}
}
public void commit2ndPhase(final TransactionContext.TransactionPhase1 changes) {
try {
if (changes == null)
return;
if (database.getMode() == PaginatedFile.MODE.READ_ONLY)
throw new TransactionException("Cannot commit changes because the database is open in read-only mode");
if (status != STATUS.COMMIT_1ST_PHASE)
throw new TransactionException("Cannot execute 2nd phase commit without having started the 1st phase");
status = STATUS.COMMIT_2ND_PHASE;
if (changes.result != null)
// WRITE TO THE WAL FIRST
database.getTransactionManager().writeTransactionToWAL(changes.modifiedPages, walFlush, txId, changes.result);
// AT THIS POINT, LOCK + VERSION CHECK, THERE IS NO NEED TO MANAGE ROLLBACK BECAUSE THERE CANNOT BE CONCURRENT TX THAT UPDATE THE SAME PAGE CONCURRENTLY
// UPDATE PAGE COUNTER FIRST
LogManager.instance()
.log(this, Level.FINE, "TX committing pages newPages=%s modifiedPages=%s (threadId=%d)", newPages, modifiedPages, Thread.currentThread().getId());
database.getPageManager().updatePages(newPages, modifiedPages, asyncFlush);
if (newPages != null) {
for (final Map.Entry entry : newPageCounters.entrySet()) {
database.getSchema().getFileById(entry.getKey()).setPageCount(entry.getValue());
database.getFileManager()
.setVirtualFileSize(entry.getKey(), (long) entry.getValue() * database.getFileManager().getFile(entry.getKey()).getPageSize());
}
}
for (final Record r : modifiedRecordsCache.values())
((RecordInternal) r).unsetDirty();
for (final int fileId : lockedFiles) {
final PaginatedComponent file = database.getSchema().getFileByIdIfExists(fileId);
if (file != null)
// THE FILE COULD BE NULL IN CASE OF INDEX COMPACTION
file.onAfterCommit();
}
} catch (final ConcurrentModificationException e) {
throw e;
} catch (final Exception e) {
LogManager.instance().log(this, Level.FINE, "Unknown exception during commit (threadId=%d)", e, Thread.currentThread().getId());
throw new TransactionException("Transaction error on commit", e);
} finally {
reset();
}
}
public void addIndexOperation(final IndexInternal index, final boolean addOperation, final Object[] keys, final RID rid) {
indexChanges.addIndexKeyLock(index, addOperation, keys, rid);
}
@Override
public boolean isAsyncFlush() {
return asyncFlush;
}
@Override
public void setAsyncFlush(final boolean value) {
this.asyncFlush = value;
}
public void reset() {
status = STATUS.INACTIVE;
if (lockedFiles != null) {
database.getTransactionManager().unlockFilesInOrder(lockedFiles);
lockedFiles = null;
}
indexChanges.reset();
modifiedPages = null;
newPages = null;
updatedRecords = null;
newPageCounters.clear();
modifiedRecordsCache.clear();
immutableRecordsCache.clear();
immutablePages.clear();
txId = -1;
}
public void removePagesOfFile(final int fileId) {
if (newPages != null)
newPages.values().removeIf(mutablePage -> fileId == mutablePage.getPageId().getFileId());
newPageCounters.remove(fileId);
if (modifiedPages != null)
modifiedPages.values().removeIf(mutablePage -> fileId == mutablePage.getPageId().getFileId());
immutablePages.values().removeIf(page -> fileId == page.getPageId().getFileId());
// IMMUTABLE RECORD, AVOID IT'S POINTING TO THE OLD OFFSET IN A MODIFIED PAGE
// SAME PAGE, REMOVE IT
immutableRecordsCache.values().removeIf(r -> r.getIdentity().getBucketId() == fileId);
if (lockedFiles != null)
lockedFiles.remove(fileId);
final PaginatedComponent component = database.getSchema().getFileByIdIfExists(fileId);
if (component instanceof LSMTreeIndexAbstract)
indexChanges.removeIndex(component.getName());
}
public TransactionIndexContext getIndexChanges() {
return indexChanges;
}
public STATUS getStatus() {
return status;
}
public void setStatus(final STATUS status) {
this.status = status;
}
private List lockFilesInOrder() {
final Set modifiedFiles = new HashSet<>();
for (final PageId p : modifiedPages.keySet())
modifiedFiles.add(p.getFileId());
if (newPages != null)
for (final PageId p : newPages.keySet())
modifiedFiles.add(p.getFileId());
indexChanges.addFilesToLock(modifiedFiles);
modifiedFiles.addAll(newPageCounters.keySet());
final long timeout = database.getConfiguration().getValueAsLong(GlobalConfiguration.COMMIT_LOCK_TIMEOUT);
final List locked = database.getTransactionManager().tryLockFiles(modifiedFiles, timeout);
// CHECK IF ALL THE LOCKED FILES STILL EXIST. FILE MISSING CAN HAPPEN IN CASE OF INDEX COMPACTION OR DROP OF A BUCKET OR AN INDEX
for (Integer f : locked)
if (!database.getFileManager().existsFile(f)) {
// ONE FILE HAS BEEN REMOVED
database.getTransactionManager().unlockFilesInOrder(locked);
rollback();
throw new ConcurrentModificationException("File with id '" + f + "' has been removed");
}
return locked;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy