/*
* DBeaver - Universal Database Manager
* Copyright (C) 2010-2017 Serge Rider (serge@jkiss.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.
*/
package org.jkiss.dbeaver.ext.postgresql.model;
import org.jkiss.code.NotNull;
import org.jkiss.code.Nullable;
import org.jkiss.dbeaver.DBException;
import org.jkiss.dbeaver.model.DBPEvaluationContext;
import org.jkiss.dbeaver.model.DBPQualifiedObject;
import org.jkiss.dbeaver.model.DBUtils;
import org.jkiss.dbeaver.model.exec.jdbc.JDBCSession;
import org.jkiss.dbeaver.model.impl.jdbc.JDBCUtils;
import org.jkiss.dbeaver.model.meta.Property;
import org.jkiss.dbeaver.model.runtime.DBRProgressMonitor;
import org.jkiss.dbeaver.model.sql.SQLUtils;
import org.jkiss.dbeaver.model.struct.DBSActionTiming;
import org.jkiss.dbeaver.model.struct.rdb.DBSManipulationType;
import org.jkiss.dbeaver.model.struct.rdb.DBSTrigger;
import org.jkiss.utils.CommonUtils;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
/**
* PostgreTrigger
*/
public class PostgreTrigger implements DBSTrigger, DBPQualifiedObject, PostgreObject, PostgreScriptObject
{
/* Bits within tgtype */
public static final int TRIGGER_TYPE_ROW = (1 << 0);
public static final int TRIGGER_TYPE_BEFORE = (1 << 1);
public static final int TRIGGER_TYPE_INSERT = (1 << 2);
public static final int TRIGGER_TYPE_DELETE = (1 << 3);
public static final int TRIGGER_TYPE_UPDATE = (1 << 4);
public static final int TRIGGER_TYPE_TRUNCATE = (1 << 5);
public static final int TRIGGER_TYPE_INSTEAD = (1 << 6);
private PostgreTableBase table;
private long objectId;
private String whenExpression;
private long functionSchemaId;
private long functionId;
private String body;
protected String name;
private DBSActionTiming actionTiming;
private DBSManipulationType[] manipulationTypes;
private PostgreTriggerType type;
private boolean persisted;
public PostgreTrigger(
PostgreTableBase table,
ResultSet dbResult)
{
this.persisted = true;
this.name = JDBCUtils.safeGetString(dbResult, "tgname");
this.table = table;
this.objectId = JDBCUtils.safeGetLong(dbResult, "oid");
this.whenExpression = JDBCUtils.safeGetString(dbResult, "tgqual");
// Get procedure
this.functionSchemaId = JDBCUtils.safeGetLong(dbResult, "func_schema_id");
this.functionId = JDBCUtils.safeGetLong(dbResult, "tgfoid");
// Parse trigger type bits
int tgType = JDBCUtils.safeGetInt(dbResult, "tgtype");
if (CommonUtils.isBitSet(tgType, TRIGGER_TYPE_BEFORE)) {
actionTiming = DBSActionTiming.BEFORE;
} else if (CommonUtils.isBitSet(tgType, TRIGGER_TYPE_INSTEAD)) {
actionTiming = DBSActionTiming.INSTEAD;
} else {
actionTiming = DBSActionTiming.AFTER;
}
List<DBSManipulationType> mt = new ArrayList<>(1);
if (CommonUtils.isBitSet(tgType, TRIGGER_TYPE_INSERT)) {
mt.add(DBSManipulationType.INSERT);
}
if (CommonUtils.isBitSet(tgType, TRIGGER_TYPE_DELETE)) {
mt.add(DBSManipulationType.DELETE);
}
if (CommonUtils.isBitSet(tgType, TRIGGER_TYPE_UPDATE)) {
mt.add(DBSManipulationType.UPDATE);
}
if (CommonUtils.isBitSet(tgType, TRIGGER_TYPE_TRUNCATE)) {
mt.add(DBSManipulationType.TRUNCATE);
}
this.manipulationTypes = mt.toArray(new DBSManipulationType[mt.size()]);
if (CommonUtils.isBitSet(tgType, TRIGGER_TYPE_ROW)) {
type = PostgreTriggerType.ROW;
} else {
type = PostgreTriggerType.STATEMENT;
}
}
@NotNull
@Override
public String getName() {
return name;
}
@Property(viewable = true, order = 2)
public DBSActionTiming getActionTiming()
{
return actionTiming;
}
@Property(viewable = true, order = 3)
public DBSManipulationType[] getManipulationTypes() {
return manipulationTypes;
}
@Property(viewable = true, order = 4)
public PostgreTriggerType getType() {
return type;
}
@Override
public boolean isPersisted() {
return persisted;
}
@Override
public PostgreTableBase getTable()
{
return table;
}
@Override
@Property(viewable = true, order = 5)
public long getObjectId() {
return objectId;
}
@Property(viewable = true, order = 6)
public String getWhenExpression()
{
return whenExpression;
}
@Property(viewable = true, order = 7)
public PostgreProcedure getFunction(DBRProgressMonitor monitor) throws DBException {
if (functionId == 0) {
return null;
}
return getDatabase().getProcedure(monitor, functionSchemaId, functionId);
}
@Nullable
@Override
public String getDescription() {
return null;
}
@Override
public PostgreTableBase getParentObject()
{
return table;
}
@NotNull
@Override
public PostgreDataSource getDataSource()
{
return table.getDataSource();
}
@NotNull
@Override
public PostgreDatabase getDatabase() {
return table.getDatabase();
}
@Override
@Property(hidden = true, editable = true, updatable = true, order = -1)
public String getObjectDefinitionText(DBRProgressMonitor monitor) throws DBException
{
if (body == null) {
try (JDBCSession session = DBUtils.openMetaSession(monitor, getDataSource(), "Read trigger definition")) {
body = JDBCUtils.queryString(session, "SELECT pg_catalog.pg_get_triggerdef(?)", objectId);
} catch (SQLException e) {
throw new DBException(e, getDataSource());
}
body = SQLUtils.formatSQL(getDataSource(), body);
}
return body;
}
@Override
public void setObjectDefinitionText(String sourceText) throws DBException
{
body = sourceText;
}
@Override
public String getFullyQualifiedName(DBPEvaluationContext context) {
return DBUtils.getFullQualifiedName(getDataSource(),
getParentObject(),
this);
}
}