/* * 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.DBUtils; import org.jkiss.dbeaver.model.exec.jdbc.JDBCPreparedStatement; import org.jkiss.dbeaver.model.exec.jdbc.JDBCResultSet; import org.jkiss.dbeaver.model.exec.jdbc.JDBCSession; import org.jkiss.dbeaver.model.exec.jdbc.JDBCStatement; import org.jkiss.dbeaver.model.impl.jdbc.JDBCUtils; import org.jkiss.dbeaver.model.impl.jdbc.cache.JDBCObjectCache; import org.jkiss.dbeaver.model.meta.Association; import org.jkiss.dbeaver.model.meta.Property; import org.jkiss.dbeaver.model.runtime.DBRProgressMonitor; import org.jkiss.dbeaver.model.struct.DBSObject; import java.sql.ResultSet; import java.sql.SQLException; import java.util.*; /** * PostgreRole */ public class PostgreRole implements PostgreObject, PostgrePermissionsOwner { public static final String CAT_SETTINGS = "Settings"; public static final String CAT_FLAGS = "Flags"; private final PostgreDatabase database; private long oid; private String name; private boolean superUser; private boolean inherit; private boolean createRole; private boolean createDatabase; private boolean canLogin; private boolean replication; private boolean bypassRls; private int connLimit; private String password; private Date validUntil; private MembersCache membersCache = new MembersCache(true); private MembersCache belongsCache = new MembersCache(false); static class MembersCache extends JDBCObjectCache<PostgreRole, PostgreRoleMember> { private final boolean members; public MembersCache(boolean members) { this.members = members; } @Override protected JDBCStatement prepareObjectsStatement(@NotNull JDBCSession session, @NotNull PostgreRole owner) throws SQLException { JDBCPreparedStatement dbStat = session.prepareStatement( "SELECT * FROM pg_catalog.pg_auth_members WHERE " + (members ? "roleid" : "member")+ "=?"); dbStat.setLong(1, owner.getObjectId()); return dbStat; } @Override protected PostgreRoleMember fetchObject(@NotNull JDBCSession session, @NotNull PostgreRole owner, @NotNull JDBCResultSet dbResult) throws SQLException, DBException { return new PostgreRoleMember(owner, dbResult); } } public PostgreRole(PostgreDatabase database, ResultSet dbResult) throws SQLException { this.database = database; this.loadInfo(dbResult); } private void loadInfo(ResultSet dbResult) throws SQLException { this.oid = JDBCUtils.safeGetLong(dbResult, "oid"); this.name = JDBCUtils.safeGetString(dbResult, "rolname"); this.superUser = JDBCUtils.safeGetBoolean(dbResult, "rolsuper"); this.inherit = JDBCUtils.safeGetBoolean(dbResult, "rolinherit"); this.createRole = JDBCUtils.safeGetBoolean(dbResult, "rolcreaterole"); this.createDatabase = JDBCUtils.safeGetBoolean(dbResult, "rolcreatedb"); this.canLogin = JDBCUtils.safeGetBoolean(dbResult, "rolcanlogin"); this.replication = JDBCUtils.safeGetBoolean(dbResult, "rolreplication"); this.bypassRls = JDBCUtils.safeGetBoolean(dbResult, "rolbypassrls"); this.connLimit = JDBCUtils.safeGetInt(dbResult, "rolconnlimit"); this.password = JDBCUtils.safeGetString(dbResult, "rolpassword"); this.validUntil = JDBCUtils.safeGetTimestamp(dbResult, "rolvaliduntil"); } @Nullable @Override public String getDescription() { return null; } @Nullable @Override public DBSObject getParentObject() { return database; } @NotNull @Override public PostgreDataSource getDataSource() { return database.getDataSource(); } @Override public boolean isPersisted() { return true; } @NotNull @Override @Property(viewable = true, order = 1) public String getName() { return name; } @NotNull @Override public PostgreDatabase getDatabase() { return database; } @Property(viewable = true, order = 2) @Override public long getObjectId() { return oid; } @Property(category = CAT_FLAGS, order = 10) public boolean isSuperUser() { return superUser; } @Property(category = CAT_FLAGS, order = 11) public boolean isInherit() { return inherit; } @Property(category = CAT_FLAGS, order = 12) public boolean isCreateRole() { return createRole; } @Property(category = CAT_FLAGS, order = 13) public boolean isCreateDatabase() { return createDatabase; } @Property(category = CAT_FLAGS, order = 14) public boolean isCanLogin() { return canLogin; } @Property(category = CAT_FLAGS, order = 15) public boolean isReplication() { return replication; } @Property(category = CAT_FLAGS, order = 16) public boolean isBypassRls() { return bypassRls; } @Property(category = CAT_SETTINGS, order = 20) public int getConnLimit() { return connLimit; } @Property(hidden = true, category = CAT_SETTINGS, order = 21) public String getPassword() { return password; } @Property(category = CAT_SETTINGS, order = 22) public Date getValidUntil() { return validUntil; } @Association public Collection<PostgreRoleMember> getMembers(DBRProgressMonitor monitor) throws DBException { return membersCache.getAllObjects(monitor, this); } @Association public Collection<PostgreRoleMember> getBelongs(DBRProgressMonitor monitor) throws DBException { return belongsCache.getAllObjects(monitor, this); } @Override public Collection<PostgrePermission> getPermissions(DBRProgressMonitor monitor) throws DBException { try (JDBCSession session = DBUtils.openMetaSession(monitor, getDataSource(), "Read role privileges")) { try (JDBCPreparedStatement dbStat = session.prepareStatement( "SELECT * FROM information_schema.table_privileges WHERE table_catalog=? AND grantee=?")) { dbStat.setString(1, getDatabase().getName()); dbStat.setString(2, getName()); try (JDBCResultSet dbResult = dbStat.executeQuery()) { Map<String, List<PostgrePrivilege>> privs = new LinkedHashMap<>(); while (dbResult.next()) { PostgrePrivilege privilege = new PostgrePrivilege(dbResult); String tableId = privilege.getTableSchema() + "." + privilege.getTableName(); List<PostgrePrivilege> privList = privs.get(tableId); if (privList == null) { privList = new ArrayList<>(); privs.put(tableId, privList); } privList.add(privilege); } // Pack to permission list List<PostgrePermission> result = new ArrayList<>(privs.size()); for (List<PostgrePrivilege> priv : privs.values()) { result.add(new PostgreRolePermission(this, priv.get(0).getTableSchema(), priv.get(0).getTableName(), priv)); } Collections.sort(result); return result; } } catch (SQLException e) { throw new DBException(e, getDataSource()); } } } @Override public String toString() { return getName(); } }