
io.questdb.cairo.CairoException Maven / Gradle / Ivy
/*******************************************************************************
* ___ _ ____ ____
* / _ \ _ _ ___ ___| |_| _ \| __ )
* | | | | | | |/ _ \/ __| __| | | | _ \
* | |_| | |_| | __/\__ \ |_| |_| | |_) |
* \__\_\\__,_|\___||___/\__|____/|____/
*
* Copyright (c) 2014-2019 Appsicle
* Copyright (c) 2019-2023 QuestDB
*
* 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 io.questdb.cairo;
import io.questdb.std.FlyweightMessageContainer;
import io.questdb.std.Os;
import io.questdb.std.Sinkable;
import io.questdb.std.ThreadLocal;
import io.questdb.std.datetime.microtime.TimestampFormatUtils;
import io.questdb.std.str.CharSink;
import io.questdb.std.str.StringSink;
import org.jetbrains.annotations.NotNull;
public class CairoException extends RuntimeException implements Sinkable, FlyweightMessageContainer {
public static final int ERRNO_FILE_DOES_NOT_EXIST = 2;
public static final int ERRNO_FILE_DOES_NOT_EXIST_WIN = 3;
public static final int ILLEGAL_OPERATION = -101;
public static final int METADATA_VALIDATION = -100;
public static final int NON_CRITICAL = -1;
private static final StackTraceElement[] EMPTY_STACK_TRACE = {};
private static final int ERRNO_ACCESS_DENIED_WIN = 5;
private static final int TABLE_DROPPED = -102;
private static final ThreadLocal tlException = new ThreadLocal<>(CairoException::new);
protected final StringSink message = new StringSink();
protected int errno;
private boolean cacheable;
private boolean interruption; // used when a query times out
private int messagePosition;
public static CairoException critical(int errno) {
CairoException ex = tlException.get();
// This is to have correct stack trace in local debugging with -ea option
assert (ex = new CairoException()) != null;
ex.message.clear();
ex.errno = errno;
ex.cacheable = false;
ex.interruption = false;
return ex;
}
public static CairoException detachedColumnMetadataMismatch(int columnIndex, CharSequence columnName, CharSequence attribute) {
return critical(METADATA_VALIDATION)
.put("Detached column [index=")
.put(columnIndex)
.put(", name=")
.put(columnName)
.put(", attribute=")
.put(attribute)
.put("] does not match current table metadata");
}
public static CairoException detachedMetadataMismatch(CharSequence attribute) {
return critical(METADATA_VALIDATION)
.put("Detached partition metadata [")
.put(attribute)
.put("] is not compatible with current table metadata");
}
public static CairoException duplicateColumn(CharSequence columnName, CharSequence columnAlias) {
CairoException exception = invalidMetadata("Duplicate column", columnName);
if (columnAlias != null) {
exception.put(", [alias=").put(columnAlias).put(']');
}
return exception;
}
public static CairoException duplicateColumn(CharSequence columnName) {
return duplicateColumn(columnName, null);
}
public static boolean errnoReadPathDoesNotExist(int errno) {
return errnoRemovePathDoesNotExist(errno) || (Os.type == Os.WINDOWS && errno == ERRNO_ACCESS_DENIED_WIN);
}
public static boolean errnoRemovePathDoesNotExist(int errno) {
return errno == ERRNO_FILE_DOES_NOT_EXIST || (Os.type == Os.WINDOWS && errno == ERRNO_FILE_DOES_NOT_EXIST_WIN);
}
public static CairoException invalidMetadata(@NotNull CharSequence msg, @NotNull CharSequence columnName) {
return critical(METADATA_VALIDATION).put(msg).put(" [name=").put(columnName).put(']');
}
public static CairoException nonCritical() {
return critical(NON_CRITICAL);
}
public static CairoException queryCancelled(int fd) {
return nonCritical().put("cancelling statement due to user request [fd=").put(fd).put(']').setInterruption(true);
}
public static CairoException queryTimedOut(int fd) {
return nonCritical().put("timeout, query aborted [fd=").put(fd).put(']').setInterruption(true);
}
public static CairoException tableDoesNotExist(CharSequence tableName) {
return nonCritical().put("table does not exist [table=").put(tableName).put(']');
}
public static CairoException tableDropped(TableToken tableToken) {
return critical(TABLE_DROPPED)
.put("table is dropped [dirName=").put(tableToken.getDirName())
.put(", tableName=").put(tableToken.getTableName())
.put(']');
}
public boolean errnoReadPathDoesNotExist() {
return errnoReadPathDoesNotExist(errno);
}
public int getErrno() {
return errno;
}
@Override
public CharSequence getFlyweightMessage() {
return message;
}
@Override
public String getMessage() {
return "[" + errno + "] " + message;
}
@Override
public int getPosition() {
return messagePosition;
}
@Override
public StackTraceElement[] getStackTrace() {
StackTraceElement[] result = EMPTY_STACK_TRACE;
// This is to have correct stack trace reported in CI
assert (result = super.getStackTrace()) != null;
return result;
}
public boolean isCacheable() {
return cacheable;
}
public boolean isCritical() {
return errno != NON_CRITICAL;
}
public boolean isInterruption() {
return interruption;
}
public boolean isTableDropped() {
return errno == TABLE_DROPPED;
}
// logged and skipped by WAL applying code
public boolean isWALTolerable() {
return !isCritical() || errno == METADATA_VALIDATION;
}
public CairoException position(int position) {
this.messagePosition = position;
return this;
}
public CairoException put(long value) {
message.put(value);
return this;
}
public CairoException put(CharSequence cs) {
message.put(cs);
return this;
}
public CairoException put(Sinkable sinkable) {
sinkable.toSink(message);
return this;
}
public CairoException put(char c) {
message.put(c);
return this;
}
public CairoException put(boolean value) {
message.put(value);
return this;
}
public CairoException putAsPrintable(CharSequence nonPrintable) {
message.putAsPrintable(nonPrintable);
return this;
}
public CairoException setCacheable(boolean cacheable) {
this.cacheable = cacheable;
return this;
}
public CairoException setInterruption(boolean interruption) {
this.interruption = interruption;
return this;
}
@Override
public void toSink(CharSink sink) {
sink.put('[').put(errno).put("]: ").put(message);
}
public CairoException ts(long timestamp) {
TimestampFormatUtils.appendDateTime(message, timestamp);
return this;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy