8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-25 02:03:03 +01:00
firebird-mirror/src/jrd/grant.epp

1319 lines
33 KiB
Plaintext
Raw Normal View History

2001-05-23 15:26:42 +02:00
/*
* PROGRAM: JRD Access Method
* MODULE: grant.epp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: SQL Grant/Revoke Handler
*
* The contents of this file are subject to the Interbase Public
* License Version 1.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.Inprise.com/IPL.html
*
* Software distributed under the License is distributed on an
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
* or implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code was created by Inprise Corporation
* and its predecessors. Portions created by Inprise Corporation are
* Copyright (C) Inprise Corporation.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
2002-10-30 07:40:58 +01:00
*
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
*
2001-05-23 15:26:42 +02:00
*/
#include "firebird.h"
2001-05-23 15:26:42 +02:00
#include <stdio.h>
#include <string.h>
2003-11-08 17:40:17 +01:00
#include "../jrd/y_ref.h"
#include "../jrd/ibase.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/jrd.h"
#include "../jrd/scl.h"
#include "../jrd/acl.h"
#include "../jrd/irq.h"
#include "../jrd/blb.h"
2003-11-30 22:04:18 +01:00
#include "../jrd/btr.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/req.h"
#include "../jrd/tra.h"
#include "../jrd/val.h"
#include "../jrd/met.h"
#include "../jrd/intl.h"
#include "../jrd/all_proto.h"
#include "../jrd/blb_proto.h"
#include "../jrd/cmp_proto.h"
#include "../jrd/dfw_proto.h"
#include "../jrd/dpm_proto.h"
#include "../jrd/dyn_proto.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/err_proto.h"
#include "../jrd/exe_proto.h"
#include "../jrd/gds_proto.h"
#include "../jrd/grant_proto.h"
#include "../jrd/jrd_proto.h"
#include "../jrd/met_proto.h"
#include "../jrd/scl_proto.h"
#include "../jrd/thd_proto.h"
2003-12-31 06:36:12 +01:00
#include "../common/utils_proto.h"
2001-05-23 15:26:42 +02:00
/* privileges given to the owner of a relation */
#define OWNER_PRIVS SCL_control | SCL_read | SCL_write | SCL_delete | SCL_protect
#define VIEW_PRIVS SCL_read | SCL_write | SCL_delete
#define ACL_BUFFER_SIZE 4096
#define DEFAULT_CLASS "SQL$DEFAULT"
#define CHECK_ACL_BOUND(to, start, length_ptr, move_length)\
{\
if (((start)->str_data + *length_ptr) < (to + move_length)) {start = GRANT_realloc_acl(start, &to, length_ptr);} \
}
#define CHECK_AND_MOVE(to, from, start, length_ptr) {CHECK_ACL_BOUND (to, start, length_ptr, 1); *(to)++ = from;}
#define CHECK_MOVE_INCR(to, from, start, length_ptr) {CHECK_ACL_BOUND (to, start, length_ptr, 1); *(to)++ = (from)++;}
DATABASE DB = STATIC "yachts.lnk";
static void define_default_class(thread_db*, const TEXT*, TEXT*, const UCHAR*, USHORT);
#ifdef NOT_USED_OR_REPLACED
static void delete_security_class(thread_db*, TEXT *);
#endif
2001-05-23 15:26:42 +02:00
static void finish_security_class(UCHAR **, USHORT, STR *, ULONG *);
static void get_object_info(thread_db*, const TEXT*, SSHORT,
2003-12-31 06:36:12 +01:00
TEXT*, TEXT*, TEXT*, bool*);
static USHORT get_public_privs(thread_db*, TEXT *, SSHORT);
static void get_user_privs(thread_db*, UCHAR **, TEXT *,
2001-05-23 15:26:42 +02:00
SSHORT, TEXT *, USHORT, STR *, ULONG *);
static void grant_user(UCHAR**, TEXT*, SSHORT, USHORT, STR*, ULONG*);
#ifdef NOT_USED_OR_REPLACED
2001-05-23 15:26:42 +02:00
static void grant_views(UCHAR **, USHORT, STR *, ULONG *);
static void purge_default_class(TEXT *, SSHORT);
#endif
static USHORT save_field_privileges(thread_db*, STR *, UCHAR **,
2001-05-23 15:26:42 +02:00
TEXT *, TEXT *, USHORT, ULONG *);
static void save_security_class(thread_db*, const TEXT*, const UCHAR*, USHORT);
2001-05-23 15:26:42 +02:00
static USHORT trans_sql_priv(TEXT *);
static SLONG squeeze_acl(UCHAR *, UCHAR **, TEXT *, SSHORT);
static bool check_string(const TEXT*, const TEXT*);
2001-05-23 15:26:42 +02:00
STR GRANT_realloc_acl(STR start_ptr,
UCHAR** write_ptr,
ULONG * buffer_length)
{
/**************************************
*
* G R A N T _ r e a l l o c _ a c l
*
**************************************
*
* Functional description
* The ACl list is greater than the current length, Increase it by
* MAX_AXL_SIZE. Do a ERR_punt in case of no memory!
*
**************************************/
const ULONG old_offset = (ULONG) (*write_ptr - start_ptr->str_data);
const ULONG realloc_length = *buffer_length + ACL_BUFFER_SIZE;
2001-05-23 15:26:42 +02:00
/* realloc the new length, ERR_punt incase of no memory */
// plb::ALL_extend((BLK*)&start_ptr, realloc_length);
2001-12-24 03:51:06 +01:00
str::extend(start_ptr, realloc_length);
2001-05-23 15:26:42 +02:00
/* the write_ptr is set back to the same offset in the new buffer*/
*write_ptr = start_ptr->str_data + old_offset;
*buffer_length = realloc_length;
return (start_ptr);
}
bool GRANT_privileges( thread_db* tdbb, SSHORT phase, DeferredWork* work,
2003-12-31 06:36:12 +01:00
jrd_tra*) // unused param, makes dfw.epp happy
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* G R A N T _ p r i v i l e g e s
*
**************************************
*
* Functional description
* Compute access control list from SQL privileges.
* This calculation is tricky and involves interaction between
* the relation-level and field-level privileges. Do not change
* the order of operations lightly.
*
**************************************/
switch (phase) {
case 1:
case 2:
2003-12-31 06:36:12 +01:00
return true;
2001-05-23 15:26:42 +02:00
case 3:
{
UCHAR *acl, *temp_acl;
UCHAR *default_acl;
2003-12-31 06:36:12 +01:00
USHORT public_priv, aggregate_public;
2001-05-23 15:26:42 +02:00
ULONG length = ACL_BUFFER_SIZE, *length_ptr = &length;
2003-12-31 06:36:12 +01:00
ULONG default_length = ACL_BUFFER_SIZE;
ULONG* default_length_ptr = &default_length;
2001-05-23 15:26:42 +02:00
SET_TDBB(tdbb);
2003-12-31 06:36:12 +01:00
bool restrct = false;
2001-05-23 15:26:42 +02:00
2004-03-07 08:58:55 +01:00
Database* dbb = tdbb->tdbb_database;
#ifdef V4_THREADING
2001-05-23 15:26:42 +02:00
V4_JRD_MUTEX_LOCK(dbb->dbb_mutexes + DBB_MUTX_grant_priv);
#endif
2003-12-31 06:36:12 +01:00
TEXT s_class[32], owner[32], default_class[32];
bool view; // unused after being retrieved.
2001-05-23 15:26:42 +02:00
get_object_info(tdbb, work->dfw_name, work->dfw_id, owner,
s_class, default_class, &view);
if (!s_class[0]) {
#ifdef V4_THREADING
2001-05-23 15:26:42 +02:00
V4_JRD_MUTEX_UNLOCK(dbb->dbb_mutexes + DBB_MUTX_grant_priv);
#endif
2003-12-31 06:36:12 +01:00
return false;
2001-05-23 15:26:42 +02:00
}
/* start the acl off by giving the owner all privileges */
2003-12-31 06:36:12 +01:00
str* str_buffer = NULL;
str* str_default_buffer = NULL;
2001-05-23 15:26:42 +02:00
2001-12-24 03:51:06 +01:00
try {
2001-05-23 15:26:42 +02:00
str_buffer = FB_NEW_RPT(*dbb->dbb_permanent, ACL_BUFFER_SIZE) str;
str_default_buffer = FB_NEW_RPT(*dbb->dbb_permanent, ACL_BUFFER_SIZE) str;
str_buffer->str_length = str_default_buffer->str_length = ACL_BUFFER_SIZE - 1;
2001-05-23 15:26:42 +02:00
acl = str_buffer->str_data;
CHECK_AND_MOVE(acl, ACL_version, str_buffer, length_ptr);
grant_user(&acl,
owner,
obj_user,
(USHORT)((work->dfw_id == obj_procedure) ?
(SCL_execute | OWNER_PRIVS) : OWNER_PRIVS),
&str_buffer,
length_ptr);
/* Pick up any relation-level privileges */
public_priv = get_public_privs(tdbb, work->dfw_name, work->dfw_id);
2001-05-23 15:26:42 +02:00
get_user_privs(tdbb, &acl, work->dfw_name, work->dfw_id,
owner, public_priv, &str_buffer, length_ptr);
2001-05-23 15:26:42 +02:00
2001-12-24 03:51:06 +01:00
if (work->dfw_id == obj_relation)
{
2001-05-23 15:26:42 +02:00
/* If we have the space to copy the acl list
no need to realloc */
if (length > default_length) {
//plb::ALL_extend((BLK*)&str_default_buffer, length);
2001-12-24 03:51:06 +01:00
str::extend(str_default_buffer, length);
2001-05-23 15:26:42 +02:00
default_length = length;
}
/* Now handle field-level privileges. This might require adding
UPDATE privilege to the relation-level acl, Therefore, save
off the relation acl because we need to add a default field
acl in that case. */
MOVE_FAST(str_buffer->str_data, str_default_buffer->str_data,
(int) (acl - str_buffer->str_data));
default_acl =
str_default_buffer->str_data + (acl -
str_buffer->str_data);
temp_acl = acl;
aggregate_public = save_field_privileges(tdbb, &str_buffer,
&acl, work->dfw_name,
owner, public_priv,
2001-05-23 15:26:42 +02:00
length_ptr);
/* SQL tables don't need the 'all priviliges to all views' acl anymore.
This special acl was only generated for SQL. */
#ifdef NOT_USED_OR_REPLACED
2001-05-23 15:26:42 +02:00
/* grant_views (&acl, VIEW_PRIVS); */
#endif
2001-05-23 15:26:42 +02:00
/* finish off and store the security class for the relation */
finish_security_class(&acl, aggregate_public, &str_buffer,
length_ptr);
save_security_class(tdbb,
s_class,
str_buffer->str_data,
(USHORT)(acl - str_buffer->str_data));
if (temp_acl != acl) /* relation privs were added? */
2003-12-31 06:36:12 +01:00
restrct = true;
2001-05-23 15:26:42 +02:00
/* if there have been privileges added at the relation level which
need to be restricted from other fields in the relation,
update the acl for them */
if (restrct)
{
finish_security_class(&default_acl, public_priv,
2001-05-23 15:26:42 +02:00
&str_default_buffer,
default_length_ptr);
define_default_class(
tdbb,
work->dfw_name,
default_class,
str_default_buffer->str_data,
(USHORT)(default_acl - str_default_buffer->str_data));
}
}
else {
finish_security_class(&acl, public_priv, &str_buffer, length_ptr);
2001-05-23 15:26:42 +02:00
save_security_class(tdbb, s_class, str_buffer->str_data,
(USHORT)(acl - str_buffer->str_data));
}
2001-12-24 03:51:06 +01:00
} // try
2003-02-13 14:33:57 +01:00
catch (const std::exception&) {
2001-12-24 03:51:06 +01:00
delete str_buffer;
delete str_default_buffer;
throw;
2001-05-23 15:26:42 +02:00
}
2001-12-24 03:51:06 +01:00
delete str_buffer;
delete str_default_buffer;
2001-05-23 15:26:42 +02:00
#ifdef V4_THREADING
2001-05-23 15:26:42 +02:00
V4_JRD_MUTEX_UNLOCK(dbb->dbb_mutexes + DBB_MUTX_grant_priv);
#endif
2001-05-23 15:26:42 +02:00
break;
}
default:
break;
}
DFW_perform_system_work();
2003-12-31 06:36:12 +01:00
return false;
2001-05-23 15:26:42 +02:00
}
static void define_default_class(
thread_db* tdbb,
2003-12-31 06:36:12 +01:00
const TEXT* relation_name,
TEXT* default_class,
const UCHAR* buffer, USHORT length)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* d e f i n e _ d e f a u l t _ c l a s s
*
**************************************
*
* Functional description
* Update the default security class for fields
* which have not been specifically granted
* any privileges. We must grant them all
* privileges which were specifically granted
* at the relation level, but none of the
* privileges we added at the relation level
* for the purpose of accessing other fields.
*
**************************************/
SET_TDBB(tdbb);
2004-03-07 08:58:55 +01:00
Database* dbb = tdbb->tdbb_database;
2001-05-23 15:26:42 +02:00
if (!*default_class) {
2003-02-18 08:13:25 +01:00
sprintf(default_class, "%s%" QUADFORMAT "d", DEFAULT_CLASS,
2001-05-23 15:26:42 +02:00
DPM_gen_id(tdbb, MET_lookup_generator(tdbb, DEFAULT_CLASS), 0,
(SINT64) 1));
blk* request = CMP_find_request(tdbb, irq_grant7, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
REL IN RDB$RELATIONS
WITH REL.RDB$RELATION_NAME EQ relation_name
2003-12-31 06:36:12 +01:00
if (!REQUEST(irq_grant7))
REQUEST(irq_grant7) = request;
2001-05-23 15:26:42 +02:00
MODIFY REL USING
REL.RDB$DEFAULT_CLASS.NULL = FALSE;
jrd_vtof(default_class, REL.RDB$DEFAULT_CLASS,
sizeof(REL.RDB$DEFAULT_CLASS));
END_MODIFY;
END_FOR;
if (!REQUEST(irq_grant7))
REQUEST(irq_grant7) = request;
}
save_security_class(tdbb, default_class, buffer, length);
2003-12-31 06:36:12 +01:00
dsc desc;
2001-05-23 15:26:42 +02:00
desc.dsc_dtype = dtype_text;
desc.dsc_sub_type = 0;
desc.dsc_scale = 0;
desc.dsc_ttype = ttype_metadata;
desc.dsc_address = (UCHAR *) relation_name;
desc.dsc_length = strlen((char*)desc.dsc_address);
2001-05-23 15:26:42 +02:00
DFW_post_work(dbb->dbb_sys_trans, dfw_scan_relation, &desc, 0);
}
#ifdef NOT_USED_OR_REPLACED
static void delete_security_class( thread_db* tdbb, TEXT * s_class)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* d e l e t e _ s e c u r i t y _ c l a s s
*
**************************************
*
* Functional description
* Delete a security class.
*
**************************************/
SET_TDBB(tdbb);
2004-03-07 08:58:55 +01:00
Database* dbb = tdbb->tdbb_database;
2001-05-23 15:26:42 +02:00
BLK handle = NULL;
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE handle)
CLS IN RDB$SECURITY_CLASSES
WITH CLS.RDB$SECURITY_CLASS EQ s_class
ERASE CLS;
END_FOR;
CMP_release(tdbb, (jrd_req*)handle);
2001-05-23 15:26:42 +02:00
}
#endif // NOT_USED_OR_REPLACED
2001-05-23 15:26:42 +02:00
static void finish_security_class(UCHAR** acl_ptr,
USHORT public_priv,
2001-05-23 15:26:42 +02:00
STR* start_ptr,
ULONG* length_ptr)
{
/**************************************
*
* f i n i s h _ s e c u r i t y _ c l a s s
*
**************************************
*
* Functional description
* Finish off a security class, putting
* in a wildcard for any public privileges.
*
**************************************/
UCHAR* acl = *acl_ptr;
2001-05-23 15:26:42 +02:00
if (public_priv) {
2001-05-23 15:26:42 +02:00
CHECK_AND_MOVE(acl, ACL_id_list, *start_ptr, length_ptr);
SCL_move_priv(&acl, public_priv, start_ptr, length_ptr);
2001-05-23 15:26:42 +02:00
}
CHECK_AND_MOVE(acl, ACL_end, *start_ptr, length_ptr);
*acl_ptr = acl;
}
static USHORT get_public_privs(
thread_db* tdbb, TEXT * object_name, SSHORT obj_type)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* g e t _ p u b l i c _ p r i v s
*
**************************************
*
* Functional description
* Get public privileges for a particular object.
*
**************************************/
SET_TDBB(tdbb);
2004-03-07 08:58:55 +01:00
Database* dbb = tdbb->tdbb_database;
2001-05-23 15:26:42 +02:00
USHORT public_priv = 0;
2001-05-23 15:26:42 +02:00
BLK request = CMP_find_request(tdbb, irq_grant5, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
PRV IN RDB$USER_PRIVILEGES
WITH PRV.RDB$RELATION_NAME EQ object_name AND
PRV.RDB$OBJECT_TYPE EQ obj_type AND
PRV.RDB$USER EQ "PUBLIC" AND
PRV.RDB$USER_TYPE EQ obj_user AND
PRV.RDB$FIELD_NAME MISSING
if (!REQUEST(irq_grant5))
REQUEST(irq_grant5) = request;
public_priv |= trans_sql_priv(PRV.RDB$PRIVILEGE);
2001-05-23 15:26:42 +02:00
END_FOR;
if (!REQUEST(irq_grant5))
REQUEST(irq_grant5) = request;
return public_priv;
2001-05-23 15:26:42 +02:00
}
static void get_object_info(
thread_db* tdbb,
2003-12-31 06:36:12 +01:00
const TEXT* object_name,
2001-05-23 15:26:42 +02:00
SSHORT obj_type,
2003-12-31 06:36:12 +01:00
TEXT* owner,
TEXT* s_class, TEXT* default_class, bool* view)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* g e t _ o b j e c t _ i n f o
*
**************************************
*
* Functional description
* This could be done in MET_scan_relation () or MET_lookup_procedure,
* but presumably we wish to make sure the information we have is
* up-to-the-minute.
*
**************************************/
SET_TDBB(tdbb);
2004-03-07 08:58:55 +01:00
Database* dbb = tdbb->tdbb_database;
2001-05-23 15:26:42 +02:00
owner[0] = '\0';
s_class[0] = '\0';
default_class[0] = '\0';
2003-12-31 06:36:12 +01:00
*view = false;
2001-05-23 15:26:42 +02:00
if (obj_type == obj_relation) {
blk* request = CMP_find_request(tdbb, irq_grant1, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
REL IN RDB$RELATIONS WITH
REL.RDB$RELATION_NAME EQ object_name
if (!REQUEST(irq_grant1))
REQUEST(irq_grant1) = request;
2003-12-31 06:36:12 +01:00
fb_utils::fb_exact_name_limit(REL.RDB$SECURITY_CLASS, sizeof(REL.RDB$SECURITY_CLASS));
2001-05-23 15:26:42 +02:00
strcpy(s_class, REL.RDB$SECURITY_CLASS);
2003-12-31 06:36:12 +01:00
fb_utils::fb_exact_name_limit(REL.RDB$DEFAULT_CLASS, sizeof(REL.RDB$DEFAULT_CLASS));
2001-05-23 15:26:42 +02:00
strcpy(default_class, REL.RDB$DEFAULT_CLASS);
2003-12-31 06:36:12 +01:00
fb_utils::fb_exact_name_limit(REL.RDB$OWNER_NAME, sizeof(REL.RDB$OWNER_NAME));
2001-05-23 15:26:42 +02:00
strcpy(owner, REL.RDB$OWNER_NAME);
*view = !REL.RDB$VIEW_BLR.isEmpty();
2001-05-23 15:26:42 +02:00
END_FOR;
if (!REQUEST(irq_grant1))
REQUEST(irq_grant1) = request;
}
else {
blk* request = CMP_find_request(tdbb, irq_grant9, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
REL IN RDB$PROCEDURES WITH
REL.RDB$PROCEDURE_NAME EQ object_name
if (!REQUEST(irq_grant9)) REQUEST(irq_grant9) = request;
2003-12-31 06:36:12 +01:00
fb_utils::fb_exact_name_limit(REL.RDB$SECURITY_CLASS, sizeof(REL.RDB$SECURITY_CLASS));
2001-05-23 15:26:42 +02:00
strcpy(s_class, REL.RDB$SECURITY_CLASS);
strcpy(default_class, "");
2003-12-31 06:36:12 +01:00
fb_utils::fb_exact_name_limit(REL.RDB$OWNER_NAME, sizeof(REL.RDB$OWNER_NAME));
2001-05-23 15:26:42 +02:00
strcpy(owner, REL.RDB$OWNER_NAME);
2003-12-31 06:36:12 +01:00
*view = false;
2001-05-23 15:26:42 +02:00
END_FOR;
if (!REQUEST(irq_grant9))
REQUEST(irq_grant9) = request;
}
}
static void get_user_privs(thread_db* tdbb,
2001-05-23 15:26:42 +02:00
UCHAR** acl_ptr,
TEXT* object_name,
SSHORT obj_type,
TEXT* owner,
USHORT public_priv,
2001-05-23 15:26:42 +02:00
STR* start_ptr,
ULONG* length_ptr)
{
/**************************************
*
* g e t _ u s e r _ p r i v s
*
**************************************
*
* Functional description
* Get privileges for a particular object.
*
**************************************/
SET_TDBB(tdbb);
2004-03-07 08:58:55 +01:00
Database* dbb = tdbb->tdbb_database;
2001-05-23 15:26:42 +02:00
UCHAR* acl = *acl_ptr;
TEXT user[32];
2001-05-23 15:26:42 +02:00
user[0] = 0;
SSHORT user_type = -2;
USHORT priv = 0;
2001-05-23 15:26:42 +02:00
BLK request = CMP_find_request(tdbb, irq_grant2, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
PRV IN RDB$USER_PRIVILEGES
WITH PRV.RDB$RELATION_NAME EQ object_name AND
PRV.RDB$OBJECT_TYPE EQ obj_type AND
(PRV.RDB$USER NE "PUBLIC" OR PRV.RDB$USER_TYPE NE obj_user) AND
(PRV.RDB$USER NE owner OR PRV.RDB$USER_TYPE NE obj_user) AND
PRV.RDB$FIELD_NAME MISSING
SORTED BY PRV.RDB$USER, PRV.RDB$USER_TYPE
if (!REQUEST(irq_grant2))
REQUEST(irq_grant2) = request;
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
fb_utils::fb_exact_name_limit(PRV.RDB$USER, sizeof(PRV.RDB$USER));
2001-05-23 15:26:42 +02:00
if (strcmp(PRV.RDB$USER, user) || PRV.RDB$USER_TYPE != user_type)
{
if (user[0])
{
grant_user(&acl, user, user_type, priv, start_ptr, length_ptr);
}
user_type = PRV.RDB$USER_TYPE;
if (user_type == obj_user)
{
priv = public_priv;
2001-05-23 15:26:42 +02:00
}
else
{
priv = 0;
}
strcpy(user, PRV.RDB$USER);
}
priv |= trans_sql_priv(PRV.RDB$PRIVILEGE);
END_FOR;
if (!REQUEST(irq_grant2))
REQUEST(irq_grant2) = request;
if (user[0])
grant_user(&acl, user, user_type, priv, start_ptr, length_ptr);
*acl_ptr = acl;
}
static void grant_user(UCHAR** acl_ptr,
TEXT* user,
SSHORT user_type,
USHORT privs,
STR* start_ptr,
ULONG* length_ptr)
{
/**************************************
*
* g r a n t _ u s e r
*
**************************************
*
* Functional description
* Grant privileges to a particular user.
*
**************************************/
UCHAR* acl = *acl_ptr;
2001-05-23 15:26:42 +02:00
CHECK_AND_MOVE(acl, ACL_id_list, *start_ptr, length_ptr);
switch (user_type) {
case obj_user_group:
CHECK_AND_MOVE(acl, id_group, *start_ptr, length_ptr);
break;
case obj_sql_role:
CHECK_AND_MOVE(acl, id_sql_role, *start_ptr, length_ptr);
break;
case obj_user:
CHECK_AND_MOVE(acl, id_person, *start_ptr, length_ptr);
break;
case obj_procedure:
CHECK_AND_MOVE(acl, id_procedure, *start_ptr, length_ptr);
break;
case obj_trigger:
CHECK_AND_MOVE(acl, id_trigger, *start_ptr, length_ptr);
break;
case obj_view:
CHECK_AND_MOVE(acl, id_view, *start_ptr, length_ptr);
break;
default:
BUGCHECK(292); /* Illegal user_type */
}
const UCHAR length = strlen(user);
2001-05-23 15:26:42 +02:00
CHECK_AND_MOVE(acl, (UCHAR) length, *start_ptr, length_ptr);
if (length) {
CHECK_ACL_BOUND(acl, *start_ptr, length_ptr, length);
MOVE_FAST(user, acl, length);
acl += length;
}
SCL_move_priv(&acl, privs, start_ptr, length_ptr);
*acl_ptr = acl;
}
#ifdef NOT_USED_OR_REPLACED
2001-05-23 15:26:42 +02:00
static void grant_views(
UCHAR ** acl_ptr,
USHORT privs, STR * start_ptr, ULONG * length_ptr)
{
/**************************************
*
* g r a n t _ v i e w s
*
**************************************
*
* Functional description
* Grant privileges to all views.
*
**************************************/
UCHAR* acl = *acl_ptr;
2001-05-23 15:26:42 +02:00
CHECK_AND_MOVE(acl, ACL_id_list, *start_ptr, length_ptr);
CHECK_AND_MOVE(acl, id_views, *start_ptr, length_ptr);
CHECK_AND_MOVE(acl, 0, *start_ptr, length_ptr);
SCL_move_priv(&acl, privs, start_ptr, length_ptr);
*acl_ptr = acl;
}
static void purge_default_class( TEXT * object_name, SSHORT obj_type)
{
/**************************************
*
* p u r g e _ d e f a u l t _ c l a s s
*
**************************************
*
* Functional description
* Get rid of any previously defined
* default security class for this relation.
*
**************************************/
thread_db* tdbb = GET_THREAD_DATA;
2004-03-07 08:58:55 +01:00
Database* dbb = tdbb->tdbb_database;
2001-05-23 15:26:42 +02:00
BLK request = CMP_find_request(tdbb, irq_grant8, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
REL IN RDB$RELATIONS
WITH REL.RDB$RELATION_NAME EQ object_name
AND REL.RDB$DEFAULT_CLASS NOT MISSING
if (!REQUEST(irq_grant8))
REQUEST(irq_grant8) = request;
2003-12-31 06:36:12 +01:00
fb_utils::fb_exact_name_limit(REL.RDB$DEFAULT_CLASS, sizeof(REL.RDB$DEFAULT_CLASS));
2001-05-23 15:26:42 +02:00
delete_security_class(tdbb, REL.RDB$DEFAULT_CLASS);
MODIFY REL USING
REL.RDB$DEFAULT_CLASS.NULL = TRUE;
END_MODIFY;
END_FOR;
if (!REQUEST(irq_grant8))
REQUEST(irq_grant8) = request;
dsc desc;
2001-05-23 15:26:42 +02:00
desc.dsc_dtype = dtype_text;
desc.dsc_sub_type = 0;
desc.dsc_scale = 0;
desc.dsc_ttype = ttype_metadata;
desc.dsc_address = (UCHAR *) object_name;
desc.dsc_length = strlen((char*)desc.dsc_address);
2001-05-23 15:26:42 +02:00
DFW_post_work(dbb->dbb_sys_trans, dfw_scan_relation, &desc, 0);
}
#endif // NOT_USED_OR_REPLACED
2001-05-23 15:26:42 +02:00
static USHORT save_field_privileges(
thread_db* tdbb,
2001-05-23 15:26:42 +02:00
STR * str_relation_buffer,
UCHAR ** acl_ptr,
TEXT * relation_name, TEXT * owner, USHORT public_priv, ULONG * length_ptr)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* s a v e _ f i e l d _ p r i v i l e g e s
*
**************************************
*
* Functional description
* Compute the privileges for all fields within a relation.
* All fields must be given the initial relation-level privileges.
* Conversely, field-level privileges must be added to the relation
* security class to be effective.
*
**************************************/
TEXT field_name[32], user[32], s_class[32];
USHORT priv, field_public, aggregate_public;
DSC desc;
SSHORT user_type;
STR str_field_buffer = 0;
STR str_field_buffer_start = 0;
2001-05-23 15:26:42 +02:00
SET_TDBB(tdbb);
2004-03-07 08:58:55 +01:00
Database* dbb = tdbb->tdbb_database;
2001-05-23 15:26:42 +02:00
2001-12-24 03:51:06 +01:00
try {
/* initialize the field-level acl buffer to include all relation-level privs */
str_field_buffer_start = FB_NEW_RPT(*dbb->dbb_permanent, *length_ptr) str;
str_field_buffer = FB_NEW_RPT(*dbb->dbb_permanent, *length_ptr) str;
str_field_buffer_start->str_length = str_field_buffer->str_length = *length_ptr - 1;
2001-05-23 15:26:42 +02:00
ULONG field_length, start_length;
2001-05-23 15:26:42 +02:00
field_length = start_length = *length_ptr;
ULONG* field_length_ptr = &field_length;
2001-05-23 15:26:42 +02:00
STR relation_buffer = *str_relation_buffer;
2001-05-23 15:26:42 +02:00
MOVE_FAST(relation_buffer->str_data, str_field_buffer->str_data,
(int) (*acl_ptr - relation_buffer->str_data));
const int field_buffer_start_index = (int) (*acl_ptr - relation_buffer->str_data);
UCHAR* field_acl = str_field_buffer->str_data + field_buffer_start_index;
UCHAR* relation_acl = relation_buffer->str_data + field_buffer_start_index;
2001-05-23 15:26:42 +02:00
int i;
2001-05-23 15:26:42 +02:00
/* remember this starting point for subsequent fields. */
for (i = 0; field_buffer_start_index > i;
i++, str_field_buffer_start->str_data[i] =
str_field_buffer->str_data[i]);
field_name[0] = 0;
user[0] = 0;
aggregate_public = public_priv;
2001-05-23 15:26:42 +02:00
blk* request = CMP_find_request(tdbb, irq_grant6, IRQ_REQUESTS);
blk* request2 = NULL;
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
FLD IN RDB$RELATION_FIELDS CROSS
PRV IN RDB$USER_PRIVILEGES
OVER RDB$RELATION_NAME, RDB$FIELD_NAME
WITH PRV.RDB$OBJECT_TYPE EQ obj_relation AND
PRV.RDB$RELATION_NAME EQ relation_name AND
PRV.RDB$FIELD_NAME NOT MISSING AND
(PRV.RDB$USER NE owner OR PRV.RDB$USER_TYPE NE obj_user)
SORTED BY PRV.RDB$FIELD_NAME, PRV.RDB$USER
if (!REQUEST(irq_grant6))
REQUEST(irq_grant6) = request;
2003-12-31 06:36:12 +01:00
fb_utils::fb_exact_name_limit(PRV.RDB$USER, sizeof(PRV.RDB$USER));
fb_utils::fb_exact_name_limit(PRV.RDB$FIELD_NAME, sizeof(PRV.RDB$FIELD_NAME));
2001-05-23 15:26:42 +02:00
/* create a control break on field_name,user */
if (strcmp(PRV.RDB$USER, user) ||
strcmp(PRV.RDB$FIELD_NAME, field_name))
{
/* flush out information for old user */
if (user[0])
{
if (strcmp(user, "PUBLIC"))
{
const USHORT field_priv =
(USHORT)( public_priv |
2001-05-23 15:26:42 +02:00
priv |
squeeze_acl(str_field_buffer->str_data,
&field_acl,
user,
user_type));
grant_user(&field_acl,
user,
user_type,
field_priv,
&str_field_buffer,
field_length_ptr);
const USHORT relation_priv =
(USHORT)( public_priv |
2001-05-23 15:26:42 +02:00
priv |
squeeze_acl(relation_buffer->str_data,
&relation_acl,
user,
user_type));
grant_user(&relation_acl,
user,
user_type,
relation_priv,
str_relation_buffer,
length_ptr);
}
else
{
field_public = field_public | public_priv | priv;
2001-05-23 15:26:42 +02:00
}
}
/* initialize for new user */
priv = 0;
strcpy(user, PRV.RDB$USER);
user_type = PRV.RDB$USER_TYPE;
}
/* create a control break on field_name */
if (strcmp(PRV.RDB$FIELD_NAME, field_name)) {
/* finish off the last field, adding a wildcard at end, giving PUBLIC
all privileges available at the table level as well as those
granted at the field level */
if (field_name[0]) {
aggregate_public |= field_public;
finish_security_class(&field_acl,
(USHORT)(field_public | public_priv),
2001-05-23 15:26:42 +02:00
&str_field_buffer,
field_length_ptr);
save_security_class(
tdbb,
s_class,
str_field_buffer->str_data,
(USHORT)(field_acl - str_field_buffer->str_data));
}
/* initialize for new field */
strcpy(field_name, PRV.RDB$FIELD_NAME);
2003-12-31 06:36:12 +01:00
fb_utils::fb_exact_name_limit(FLD.RDB$SECURITY_CLASS, sizeof(FLD.RDB$SECURITY_CLASS));
2001-05-23 15:26:42 +02:00
strcpy(s_class, FLD.RDB$SECURITY_CLASS);
if (!s_class[0]) {
2002-07-01 17:46:07 +02:00
/* We should never get here (I think) because this
value is set by dyn.e when the rdb$user_privileges
record is stored. There's also a before store trigger
on rdb$user_privileges, but it isn't so smart. -- AWH
*/
2003-02-18 08:13:25 +01:00
sprintf(s_class, "%s%" QUADFORMAT "d", "SQL$GRANT",
2002-07-01 17:46:07 +02:00
DPM_gen_id(tdbb, MET_lookup_generator(tdbb, "RDB$SECURITY_CLASS"),
0, (SINT64) 1));
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request2)
FLD2 IN RDB$RELATION_FIELDS WITH
FLD2.RDB$RELATION_NAME EQ FLD.RDB$RELATION_NAME
AND FLD2.RDB$FIELD_NAME EQ FLD.RDB$FIELD_NAME
MODIFY FLD2
jrd_vtof(s_class, FLD2.RDB$SECURITY_CLASS,
sizeof(FLD2.RDB$SECURITY_CLASS));
END_MODIFY;
END_FOR;
}
field_public = 0;
/* restart a security class at the end of the relation-level privs */
i = 0;
while (field_buffer_start_index > i)
{
++i;
str_field_buffer->str_data[i] =
str_field_buffer_start->str_data[i];
}
field_acl = str_field_buffer->str_data + field_buffer_start_index;
}
priv |= trans_sql_priv(PRV.RDB$PRIVILEGE);
END_FOR;
if (!REQUEST(irq_grant6))
REQUEST(irq_grant6) = request;
if (request2)
CMP_release(tdbb, (jrd_req*)request2);
2001-05-23 15:26:42 +02:00
/* flush out the last user's info */
if (user[0])
{
if (strcmp(user, "PUBLIC"))
{
const USHORT field_priv =
(USHORT)( public_priv |
2001-05-23 15:26:42 +02:00
priv |
squeeze_acl(str_field_buffer->str_data,
&field_acl,
user,
user_type));
grant_user(&field_acl,
user,
user_type,
field_priv,
&str_field_buffer,
field_length_ptr);
const USHORT relation_priv =
(USHORT)( public_priv |
2001-05-23 15:26:42 +02:00
priv |
squeeze_acl(relation_buffer->str_data,
&relation_acl,
user,
user_type));
grant_user(&relation_acl,
user,
user_type,
relation_priv,
str_relation_buffer,
length_ptr);
}
else
{
field_public = field_public | public_priv | priv;
2001-05-23 15:26:42 +02:00
}
}
/* flush out the last field's info, and schedule a format update */
if (field_name[0])
{
aggregate_public |= field_public;
finish_security_class(&field_acl,
(USHORT)(field_public | public_priv),
2001-05-23 15:26:42 +02:00
&str_field_buffer,
field_length_ptr);
save_security_class(tdbb,
s_class,
str_field_buffer->str_data,
(USHORT)(field_acl - str_field_buffer->str_data));
desc.dsc_dtype = dtype_text;
desc.dsc_sub_type = 0;
desc.dsc_scale = 0;
desc.dsc_ttype = ttype_metadata;
desc.dsc_address = (UCHAR *) relation_name;
desc.dsc_length = strlen((char*)desc.dsc_address);
2001-05-23 15:26:42 +02:00
DFW_post_work(dbb->dbb_sys_trans, dfw_update_format, &desc, 0);
}
*acl_ptr = relation_acl;
}
2003-02-13 14:33:57 +01:00
catch (const std::exception&) {
2001-12-24 03:51:06 +01:00
delete str_field_buffer;
delete str_field_buffer_start;
throw;
2001-12-24 03:51:06 +01:00
}
delete str_field_buffer;
delete str_field_buffer_start;
2001-05-23 15:26:42 +02:00
return aggregate_public;
}
static void save_security_class(thread_db* tdbb,
const TEXT* s_class,
const UCHAR* buffer,
2001-05-23 15:26:42 +02:00
USHORT length)
{
/**************************************
*
* s a v e _ s e c u r i t y _ c l a s s
*
**************************************
*
* Functional description
* Store or update the named security class.
*
**************************************/
SET_TDBB(tdbb);
2004-03-07 08:58:55 +01:00
Database* dbb = tdbb->tdbb_database;
2001-05-23 15:26:42 +02:00
bid blob_id;
blb* blob = BLB_create(tdbb, dbb->dbb_sys_trans, &blob_id);
2001-05-23 15:26:42 +02:00
BLB_put_segment(tdbb, blob, buffer, length);
BLB_close(tdbb, blob);
blk* request = CMP_find_request(tdbb, irq_grant3, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
bool found = false;
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
CLS IN RDB$SECURITY_CLASSES
WITH CLS.RDB$SECURITY_CLASS EQ s_class
if (!REQUEST(irq_grant3))
REQUEST(irq_grant3) = request;
2003-12-31 06:36:12 +01:00
found = true;
2001-05-23 15:26:42 +02:00
MODIFY CLS
CLS.RDB$ACL = blob_id;
END_MODIFY;
END_FOR;
if (!REQUEST(irq_grant3))
REQUEST(irq_grant3) = request;
if (!found) {
request = CMP_find_request(tdbb, irq_grant4, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
STORE(REQUEST_HANDLE request)
CLS IN RDB$SECURITY_CLASSES
jrd_vtof(s_class, CLS.RDB$SECURITY_CLASS,
sizeof(CLS.RDB$SECURITY_CLASS));
CLS.RDB$ACL = blob_id;
END_STORE;
if (!REQUEST(irq_grant4))
REQUEST(irq_grant4) = request;
}
}
static USHORT trans_sql_priv( TEXT * privileges)
{
/**************************************
*
* t r a n s _ s q l _ p r i v
*
**************************************
*
* Functional description
* Map a SQL privilege letter into an internal privilege bit.
*
**************************************/
USHORT priv = 0;
2001-05-23 15:26:42 +02:00
switch (UPPER7(privileges[0])) {
case 'S':
priv |= SCL_read;
break;
case 'I':
priv |= SCL_sql_insert;
break;
case 'U':
priv |= SCL_sql_update;
break;
case 'D':
priv |= SCL_sql_delete;
break;
case 'R':
priv |= SCL_sql_references;
break;
case 'X':
priv |= SCL_execute;
break;
}
return priv;
}
static SLONG squeeze_acl(UCHAR* acl_base,
UCHAR** acl_ptr,
TEXT* user,
SSHORT user_type)
{
/**************************************
*
* s q u e e z e _ a c l
*
**************************************
*
* Functional description
* Walk an access control list looking for a hit. If a hit
* is found, return privileges and squeeze out that acl-element.
* The caller will use the returned privilege to insert a new
* privilege for the input user.
*
**************************************/
UCHAR *dup_acl;
UCHAR *dest, *source;
SLONG privilege = 0;
UCHAR c;
int length;
UCHAR l;
/* Make sure that this half-finished acl looks good enough to process. */
**acl_ptr = 0;
UCHAR* acl = acl_base;
2001-05-23 15:26:42 +02:00
if (*acl++ != ACL_version)
BUGCHECK(160); /* msg 160 wrong ACL version */
bool hit = false;
2001-12-24 03:51:06 +01:00
while ( (c = *acl++) )
2001-05-23 15:26:42 +02:00
switch (c) {
case ACL_id_list:
dup_acl = acl - 1;
hit = true;
2001-12-24 03:51:06 +01:00
while ( (c = *acl++) ) {
2001-05-23 15:26:42 +02:00
switch (c) {
case id_person:
if (user_type != obj_user)
hit = false;
if (check_string((TEXT*)acl, user))
hit = false;
2001-05-23 15:26:42 +02:00
break;
case id_sql_role:
if (user_type != obj_sql_role)
hit = false;
if (check_string((TEXT*)acl, user))
hit = false;
2001-05-23 15:26:42 +02:00
break;
case id_view:
if (user_type != obj_view)
hit = false;
if (check_string((TEXT*)acl, user))
hit = false;
2001-05-23 15:26:42 +02:00
break;
case id_procedure:
if (user_type != obj_procedure)
hit = false;
if (check_string((TEXT*)acl, user))
hit = false;
2001-05-23 15:26:42 +02:00
break;
case id_trigger:
if (user_type != obj_trigger)
hit = false;
if (check_string((TEXT*)acl, user))
hit = false;
2001-05-23 15:26:42 +02:00
break;
case id_project:
case id_organization:
hit = false;
// CVC: What's the idea of calling a function whose only
// result is boolean without checking it?
check_string((TEXT*)acl, user);
2001-05-23 15:26:42 +02:00
break;
case id_views:
hit = false;
2001-05-23 15:26:42 +02:00
break;
case id_node:
case id_user:
hit = false;
2001-05-23 15:26:42 +02:00
for (l = *acl++; l; acl++, l--);
break;
case id_group:
if (user_type != obj_user_group)
hit = false;
if (check_string((TEXT*)acl, user))
hit = false;
2001-05-23 15:26:42 +02:00
break;
default:
BUGCHECK(293); /* bad ACL */
}
acl += *acl + 1;
}
break;
case ACL_priv_list:
if (hit) {
2001-12-24 03:51:06 +01:00
while ( (c = *acl++) )
2001-05-23 15:26:42 +02:00
switch (c) {
case priv_control:
privilege |= SCL_control;
break;
case priv_read:
privilege |= SCL_read;
break;
case priv_write:
privilege |= SCL_write;
break;
case priv_sql_insert:
privilege |= SCL_sql_insert;
break;
case priv_sql_delete:
privilege |= SCL_sql_delete;
break;
case priv_sql_references:
privilege |= SCL_sql_references;
break;
case priv_sql_update:
privilege |= SCL_sql_update;
break;
case priv_delete:
privilege |= SCL_delete;
break;
case priv_grant:
privilege |= SCL_grant;
break;
case priv_protect:
privilege |= SCL_protect;
break;
case priv_execute:
privilege |= SCL_execute;
break;
default:
BUGCHECK(293); /* bad ACL */
}
/* Squeeze out duplicate acl element. */
dest = dup_acl;
source = acl;
length = *acl_ptr - source + 1;
*acl_ptr = *acl_ptr - (source - dest);
acl = dup_acl;
for (; length; *dest++ = *source++, length--);
}
else
while (*acl++);
break;
default:
BUGCHECK(293); /* bad ACL */
}
return privilege;
}
static bool check_string( const TEXT* acl, const TEXT* string)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* c h e c k _ s t r i n g
*
**************************************
*
* Functional description
* Check a string against an acl string. If they don't match,
* return true.
2001-05-23 15:26:42 +02:00
*
**************************************/
/* JPN: Since Kanji User names are not allowed, No need to fix this UPPER loop. */
USHORT l = *acl++;
if (l)
2001-05-23 15:26:42 +02:00
do {
const TEXT c1 = *acl++;
const TEXT c2 = *string++;
2001-05-23 15:26:42 +02:00
if (UPPER7(c1) != UPPER7(c2))
return true;
2001-05-23 15:26:42 +02:00
} while (--l);
return (*string && *string != ' ') ? true : false;
2001-05-23 15:26:42 +02:00
}