home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 5 / 05.iso / a / a025 / 5.ddi / INSTCAT.SQL < prev    next >
Encoding:
Text File  |  1993-08-24  |  98.0 KB  |  3,404 lines

  1. /* instcat.sql */
  2.  
  3. /*
  4. NOTE:  you MUST change the last row inserted into spt_server_info
  5. to be version number of this file.    the convention is jj.nn.dddd, where
  6. jj is the major version number ("01" now), nn is the minor version number
  7. ("01" now), and dddd is the date in the form of month and day (mmdd) of
  8. the date you check in this file.  Add (year-1991)*12 to the month to keep
  9. in sync with the driver version numbers, i.e. checking in on Feb 5 1993
  10. would mean setting the value to 01.01.2605.
  11. */
  12.  
  13. /****************************************************************************/
  14. /* This portion sets up the ability to perform all the functions in this    */
  15. /* script                                                                    */
  16. /****************************************************************************/
  17. use master
  18. go
  19. dump tran master with truncate_only
  20. go
  21.  
  22. if exists (select * from sysobjects
  23.        where name = 'sp_configure' and sysstat & 7 = 4)
  24. begin
  25.     execute sp_configure 'update',1
  26. end
  27. reconfigure with override
  28. go
  29.  
  30. /*
  31. ** If old versions of tables exist, drop them.
  32. */
  33. if (exists (select * from sysobjects
  34.         where name = 'MSdatatype_info' and sysstat & 7 = 3))
  35.     drop table MSdatatype_info
  36. go
  37. if (exists (select * from sysobjects
  38.         where name = 'MSdatatype_info_ext' and sysstat & 7 = 3))
  39.     drop table MSdatatype_info_ext
  40. go
  41. if (exists (select * from sysobjects
  42.         where name = 'MStable_types' and sysstat & 7 = 3))
  43.     drop table MStable_types
  44. go
  45. if (exists (select * from sysobjects
  46.         where name = 'MSserver_info' and sysstat & 7 = 3))
  47.     drop table MSserver_info
  48. go
  49. if (exists (select * from sysobjects
  50.         where name = 'spt_table_types' and sysstat & 7 = 3))
  51.     drop table spt_table_types    /* no longer used */
  52. go
  53.  
  54. /*
  55. ** If tables or procs already exist, drop them.
  56. */
  57.  
  58. if (exists (select * from sysobjects
  59.         where name = 'spt_datatype_info' and sysstat & 7 = 3))
  60.     drop table spt_datatype_info
  61. go
  62. if (exists (select * from sysobjects
  63.         where name = 'spt_datatype_info_ext' and sysstat & 7 = 3))
  64.     drop table spt_datatype_info_ext
  65. go
  66. if (exists (select * from sysobjects
  67.         where name = 'spt_server_info' and sysstat & 7 = 3))
  68.     drop table spt_server_info
  69. go
  70. if (exists (select * from sysobjects
  71.         where name = 'sp_tables' and sysstat & 7 = 4))
  72.     drop proc sp_tables
  73. go
  74. if (exists (select * from sysobjects
  75.         where name = 'sp_statistics' and sysstat & 7 = 4))
  76.     drop proc sp_statistics
  77. go
  78. if (exists (select * from sysobjects
  79.         where name = 'sp_columns' and sysstat & 7 = 4))
  80.     drop proc sp_columns
  81. go
  82. if (exists (select * from sysobjects
  83.         where name = 'sp_fkeys' and sysstat & 7 = 4))
  84.     drop proc sp_fkeys
  85. go
  86. if (exists (select * from sysobjects
  87.         where name = 'sp_pkeys' and sysstat & 7 = 4))
  88.     drop proc sp_pkeys
  89. dump tran master with truncate_only
  90. go
  91.  
  92. go
  93. if (exists (select * from sysobjects
  94.         where name = 'sp_stored_procedures' and sysstat & 7 = 4))
  95.     drop proc sp_stored_procedures
  96. go
  97. if (exists (select * from sysobjects
  98.         where name = 'sp_sproc_columns' and sysstat & 7 = 4))
  99.     drop proc sp_sproc_columns
  100. go
  101. if (exists (select * from sysobjects
  102.         where name = 'sp_table_privileges' and sysstat & 7 = 4))
  103.     drop proc sp_table_privileges
  104. go
  105. if (exists (select * from sysobjects
  106.         where name = 'sp_column_privileges' and sysstat & 7 = 4))
  107.     drop proc sp_column_privileges
  108. go
  109. if (exists (select * from sysobjects
  110.         where name = 'sp_server_info' and sysstat & 7 = 4))
  111.     drop proc sp_server_info
  112. go
  113. if (exists (select * from sysobjects
  114.         where name = 'sp_datatype_info' and sysstat & 7 = 4))
  115.     drop proc sp_datatype_info
  116. go
  117. if (exists (select * from sysobjects
  118.         where name = 'sp_special_columns' and sysstat & 7 = 4))
  119.     drop proc sp_special_columns
  120. go
  121. if (exists (select * from sysobjects
  122.         where name = 'sp_databases' and sysstat & 7 = 4))
  123.     drop proc sp_databases
  124. go
  125.  
  126. dump tran master with truncate_only
  127. go
  128.  
  129. print "creating table spt_datatype_info_ext"
  130. go
  131. create table spt_datatype_info_ext (
  132.                 user_type        smallint    not null,
  133.                 create_params    varchar(32) null)
  134. go
  135.  
  136. grant select on spt_datatype_info_ext to public
  137. go
  138.  
  139.  
  140. insert into spt_datatype_info_ext
  141.     /* CHAR      user_type, create_params */
  142.     values             (1,    "length" )
  143.  
  144. insert into spt_datatype_info_ext
  145.     /* VARCHAR     user_type, create_params */
  146.     values             (2,    "max length" )
  147.  
  148. insert into spt_datatype_info_ext
  149.     /* BINARY     user_type, create_params */
  150.     values             (3,    "length" )
  151.  
  152. insert into spt_datatype_info_ext
  153.     /* VARBINARY user_type, create_params */
  154.     values             (4,    "max length" )
  155.  
  156. insert into spt_datatype_info_ext
  157.     /* SYSNAME     user_type, create_params */
  158.     values             (18,    "max length" )
  159. go
  160.  
  161. print "creating table spt_datatype_info"
  162. go
  163. create table spt_datatype_info (
  164.                 ss_dtype           tinyint        not null,
  165.                 type_name          varchar(32)  not null,
  166.                 data_type          smallint     not null,
  167.                 data_precision     int          null,
  168.                 numeric_scale      smallint     null,
  169.                 numeric_radix      smallint     null,
  170.                 length             int          null,
  171.                 literal_prefix     varchar(32)  null,
  172.                 literal_suffix     varchar(32)  null,
  173.                 create_params      varchar(32)  null,
  174.                 nullable           smallint     not null,
  175.                 case_sensitive     smallint     not null,
  176.                 searchable         smallint     not null,
  177.                 unsigned_attribute smallint     null,
  178.                 money              smallint     not null,
  179.                 auto_increment     smallint     null,
  180.                 local_type_name    varchar(128) not null,
  181.                 aux                int          null)
  182. go
  183.  
  184. /*
  185.     There is a complicated set of SQL used to deal with
  186.     the SQL Server Null data types (MONEYn, INTn, etc.)
  187.     ISNULL is the only conditional SQL Server function that can be used
  188.     to differentiate between these types depending on size.
  189.  
  190.     The aux column in the above table is used to differentiate
  191.     the null data types from the non-null types.
  192.  
  193.     The aux column contains NULL for the null data types and 0
  194.     for the non-null data types.
  195.  
  196.     The following SQL returns the contents of the aux column (0)
  197.     for the non-null data types and returns a variable non-zero
  198.     value for the null data types.
  199.  
  200.                                  " I   I I FFMMDD"
  201.                                  " 1   2 4 484848"
  202.     isnull(d.aux,ascii(substring("666AAA@@@CB??GG",
  203.     2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)
  204.  
  205.     The '2*(d.ss_dtype%35+1)+2-8/c.length' selects a specific character of
  206.     the substring mask depending on the null data type and its size, i.e.
  207.     null MONEY4 or null MONEY8.  The character selected is then converted
  208.     to its binary value and an appropriate bias (i.e. 60) is subtracted to
  209.     return the correct non-zero value.    This value may be used as a
  210.     constant, i.e. ODBC data type, precision, scale, etc., or used as an
  211.     index with a substring to pick out a character string, i.e. type name.
  212.  
  213.     The comments above the substring mask denote which character is
  214.     selected for each null data type, i.e. In (INTn), Fn (FLOATn),
  215.     Mn (MONEYn) and Dn (DATETIMn).
  216. */
  217.  
  218. grant select on spt_datatype_info to public
  219. go
  220.  
  221. /* Get case sensitivity */
  222. if 'A' = 'A' /* create dummy begin block */
  223. begin
  224.     declare @case smallint
  225.  
  226.     select @case = 0
  227.     select @case = 1 where 'a' != 'A'
  228.  
  229.     /* Local Binary */
  230.     insert into spt_datatype_info
  231.     /* ss_type,name    ,data_type,prec,scale,rdx ,len ,prf ,suf ,cp      ,nul,case,srch,unsigned,money,auto,local   ,aux */ values
  232.     (  45     ,"binary",-2       ,null,null ,null,null,"0x",null,"length",1  ,0   ,2   ,null    ,0    ,null,"binary",0)
  233.  
  234.     /* Local Bit */
  235.     insert into spt_datatype_info
  236.     /* ss_type,name ,data_type, prec,scale,rdx,len ,prf ,suf ,cp  ,nul,case,srch,unsigned,money,auto,local,aux */ values
  237.     (  50     ,"bit",-7       ,    1   ,0    ,2  ,null,null,null,null,0  ,0   ,2   ,null    ,0    ,null,"bit",0)
  238.  
  239.     /* Local Char */
  240.     insert into spt_datatype_info
  241.     /* ss_type,name  ,data_type,prec,scale,rdx ,len ,prf,suf,cp      ,nul,case ,srch,unsigned,money,auto,local ,aux */ values
  242.     (  47     ,"char",1        ,null,null ,null,null,"'","'","length",1  ,@case,3   ,null    ,0    ,null,"char",0)
  243.  
  244.     /* Local Datetime */
  245.     insert into spt_datatype_info
  246.     /* ss_type,name      ,data_type,prec,scale,rdx,len,prf,suf,cp  ,nul,case,srch,unsigned,money,auto,local     ,aux */ values
  247.     (  61     ,"datetime",11       ,23  ,3    ,10 ,16 ,"'","'",null,1  ,0   ,3   ,null    ,0    ,null,"datetime",0)
  248.  
  249.     /* Local Smalldatetime */
  250.     insert into spt_datatype_info
  251.     /* ss_type,name           ,data_type,prec,scale,rdx,len,prf,suf,cp  ,nul,case,srch,unsigned,money,auto,local          ,aux */ values
  252.     (  58     ,"smalldatetime",11       ,16  ,0    ,10 ,16 ,"'","'",null,1  ,0   ,3   ,null    ,0    ,null,"smalldatetime",0)
  253.  
  254.     /* Local Datetimn */
  255.     insert into spt_datatype_info  /* sql server type is "datetimn" */
  256.     /* ss_type,name           ,data_type,prec,scale,rdx,len,prf,suf,cp  ,nul,case,srch,unsigned,money,auto,local     ,aux */ values
  257.     (  111    ,"smalldatetime",0        ,0   ,0    ,10 ,0  ,"'","'",null,1  ,0   ,3   ,null    ,0    ,null,"datetime",null)
  258.  
  259.     /* Local Float */
  260.     insert into spt_datatype_info
  261.     /* ss_type,name   ,data_type,prec,scale,rdx,len ,prf ,suf ,cp  ,nul,case,srch,unsigned,money,auto,local  ,aux */ values
  262.     (  62     ,"float",6        ,15  ,null ,10 ,null,null,null,null,1  ,0   ,2   ,0       ,0    ,0   ,"float",0)
  263.  
  264.     /* Local RealFloat */
  265.     insert into spt_datatype_info  /* sql server type is "floatn" */
  266.     /* ss_type,name               ,data_type,prec,scale,rdx,len,prf ,suf ,cp  ,nul,case,srch,unsigned,money,auto,local            ,aux */ values
  267.     (  109    ,"float        real",0        ,0   ,null ,10 ,0  ,null,null,null,1  ,0   ,2   ,0       ,0    ,0   ,"real      float",null)
  268.  
  269.     /* Local Real */
  270.     insert into spt_datatype_info
  271.     /* ss_type,name  ,data_type,prec,scale,rdx,len ,prf ,suf ,cp  ,nul,case,srch,unsigned,money,auto,local ,aux */ values
  272.     (  59     ,"real",7        ,7   ,null ,10 ,null,null,null,null,1  ,0   ,2   ,0       ,0    ,0   ,"real",0)
  273.  
  274.     /* Local Smallmoney */
  275.     insert into spt_datatype_info
  276.     /* ss_type,name        ,data_type,prec,scale,rdx,len ,prf,suf ,cp  ,nul,case,srch,unsigned,money,auto,local       ,aux */ values
  277.     (  122    ,"smallmoney",3        ,10  ,4    ,10 ,null,"$",null,null,1  ,0   ,2   ,0       ,1    ,0   ,"smallmoney",0)
  278.  
  279.     /* Local Int */
  280.     insert into spt_datatype_info
  281.     /* ss_type,name ,data_type,prec,scale,rdx,len ,prf ,suf ,cp  ,nul,case,srch,unsigned,money,auto,local,aux */ values
  282.     (  56     ,"int",4        ,10  ,0    ,10 ,null,null,null,null,1  ,0   ,2   ,0       ,0    ,0   ,"int",0)
  283.  
  284.     /* Local Intn */
  285.     insert into spt_datatype_info  /* sql server type is "intn" */
  286.     /* ss_type,name                  ,data_type,prec,scale,rdx,len,prf ,suf ,cp  ,nul,case,srch,unsigned,money,auto,local               ,aux */ values
  287.     (       38,"smallint     tinyint",0        ,0   ,0    ,10 ,0  ,null,null,null,1  ,0   ,2   ,0       ,0    ,0   ,"tinyint   smallint",null)
  288.  
  289.     /* Local Money */
  290.     insert into spt_datatype_info
  291.     /* ss_type,name   ,data_type,prec,scale,rdx,len ,prf,suf ,cp  ,nul,case,srch,unsigned,money,auto,local  ,aux */ values
  292.     (  60     ,"money",3        ,19  ,4    ,10 ,null,"$",null,null,1  ,0   ,2   ,0       ,1    ,0   ,"money",0)
  293.  
  294.     /* Local Moneyn */
  295.     insert into spt_datatype_info    /* sql server type is "moneyn" */
  296.     /* ss_type,name        ,data_type,prec,scale,rdx,len,prf,suf ,cp  ,nul,case,srch,unsigned,money,auto,local            ,aux */ values
  297.     (  110    ,"smallmoney",0        ,0   ,4    ,10 ,0  ,"$",null,null,1  ,0   ,2   ,0       ,1    ,0   ,"smallmoneymoney",null)
  298.  
  299.     /* Local Smallint */
  300.     insert into spt_datatype_info
  301.     /* ss_type,name      ,data_type,prec,scale,rdx,len ,prf ,suf ,cp  ,nul,case,srch,unsigned,money,auto,local     ,aux */ values
  302.     (  52     ,"smallint",5        ,5   ,0    ,10 ,null,null,null,null,1  ,0   ,2   ,0       ,0    ,0   ,"smallint",0)
  303.  
  304.     /* Local Text */
  305.     insert into spt_datatype_info
  306.     /* ss_type,name  ,data_type,prec      ,scale,rdx ,len       ,prf,suf,cp  ,nul,case ,srch,unsigned,money,auto,local ,aux */ values
  307.     (  35     ,"text",-1       ,2147483647,null ,null,2147483647,"'","'",null,1  ,@case,1   ,null    ,0    ,null,"text",0)
  308.  
  309.     /* Local Varbinary */
  310.     insert into spt_datatype_info
  311.     /* ss_type,name       ,data_type,prec,scale,rdx ,len ,prf ,suf ,cp          ,nul,case,srch,unsigned,money,auto,local      ,aux */ values
  312.     (  37     ,"varbinary",-3       ,null,null ,null,null,"0x",null,"max length",1  ,0   ,2   ,null    ,0    ,null,"varbinary",0)
  313.  
  314.     /* Local Tinyint */
  315.     insert into spt_datatype_info
  316.     /* ss_type,name     ,data_type,prec,scale,rdx,len ,prf ,suf ,cp  ,nul,case,srch,unsigned,money,auto,local    ,aux */ values
  317.     (  48     ,"tinyint",-6       ,3   ,0    ,10 ,null,null,null,null,1  ,0   ,2   ,1       ,0    ,0   ,"tinyint",0)
  318.  
  319.     /* Local Varchar */
  320.     insert into spt_datatype_info
  321.     /* ss_type,name     ,data_type,prec,scale,rdx ,len ,prf,suf,cp          ,nul,case ,srch,unsigned,money,auto,local    ,aux */ values
  322.     (  39     ,"varchar",12       ,null,null ,null,null,"'","'","max length",1  ,@case,3   ,null    ,0    ,null,"varchar",0)
  323.  
  324.     /* Local Image */
  325.     insert into spt_datatype_info
  326.     /* ss_type,name   ,data_type,prec      ,scale,rdx ,len       ,prf ,suf ,cp  ,nul,case,srch,unsigned,money,auto,local  ,aux */ values
  327.     (  34     ,"image",-4       ,2147483647,null ,null,2147483647,"0x",null,null,1  ,0   ,1   ,null    ,0    ,null,"image",0)
  328. end
  329. go
  330.  
  331. dump tran master with truncate_only
  332. go
  333.  
  334. print "creating table spt_server_info"
  335. go
  336. create table spt_server_info (
  337.               attribute_id        int,
  338.               attribute_name    varchar(60),
  339.               attribute_value    varchar(255))
  340. go
  341.  
  342. insert into spt_server_info
  343.     values (1,"DBMS_NAME","SQL Server")
  344. insert into spt_server_info
  345.     values (2,"DBMS_VER",@@version)
  346. insert into spt_server_info
  347.     values (10,"OWNER_TERM","owner")
  348. insert into spt_server_info
  349.     values (11,"TABLE_TERM","table")
  350. insert into spt_server_info
  351.     values (12,"MAX_OWNER_NAME_LENGTH","30")
  352. insert into spt_server_info
  353.     values (13,"TABLE_LENGTH","30")
  354. insert into spt_server_info
  355.     values (14,"MAX_QUAL_LENGTH","30")
  356. insert into spt_server_info
  357.     values (15,"COLUMN_LENGTH","30")
  358. insert into spt_server_info
  359.     values (16,"IDENTIFIER_CASE","MIXED")
  360. insert into spt_server_info
  361.     values (17,"TX_ISOLATION","2")
  362. go
  363. /*
  364. ** syscharsets doesn't exist on pre rel. 4.2 servers.
  365. */
  366. if not exists (select * from sysobjects where name = 'syscharsets')
  367.  if 'a' = 'A'    /* pre 4.2 case insensitive server */
  368.         insert into spt_server_info values
  369.            (18,"COLLATION_SEQ",
  370.            'charset=cp850 sort_order=caseless_34 charset_num=2 sort_order_num=49')
  371.      else            /* pre 4.2 case sensitive server */
  372.         insert into spt_server_info values
  373.            (18,"COLLATION_SEQ",
  374.             'charset=cp850 sort_order=bin_cp850 charset_num=2 sort_order_num=40')
  375. go
  376.  
  377. /*    Check if Version 1.11 */
  378. if  (charindex('1.11', @@version) != 0)
  379.    begin
  380.     print ""
  381.     print ""
  382.     print "Installing catalog procedures on a 1.x or 4.0 server:"
  383.     print "Ignore the following error messages"
  384.     print ""
  385.     print ""
  386.    end
  387. go
  388.  
  389. /*
  390. ** Use the insert that utilizes syscharsets if it exists.
  391. */
  392. if exists (select * from sysobjects where name = 'syscharsets')
  393.    insert into spt_server_info
  394.       select 18,"COLLATION_SEQ",
  395.          "charset="+t2.name+" sort_order="+t1.name
  396.          +" charset_num="+rtrim(convert(char(4),t1.csid))+
  397.          " sort_order_num="+rtrim(convert(char(4),t1.id))
  398.      from syscharsets t1, syscharsets t2, sysconfigures t3
  399.      where t1.csid=t2.id and t1.id=t3.value and t3.config=123
  400. go
  401.  
  402. insert into spt_server_info
  403.     values (19,"SAVEPOINT_SUPPORT","Y")
  404. insert into spt_server_info
  405.     values (20,"MULTI_RESULT_SETS","Y")
  406. insert into spt_server_info
  407.     values (22,"ACCESSIBLE_TABLES","Y")
  408. insert into spt_server_info
  409.     values (100,"USERID_LENGTH","30")
  410. insert into spt_server_info
  411.     values (101,"QUALIFIER_TERM","database")
  412. insert into spt_server_info
  413.     values (102,"NAMED_TRANSACTIONS","Y")
  414. insert into spt_server_info
  415.     values (103,"SPROC_AS_LANGUAGE","Y")
  416. insert into spt_server_info
  417.     values (104,"ACCESSIBLE_SPROC","Y")
  418. insert into spt_server_info
  419.     values (105,"MAX_INDEX_COLS","16")
  420. insert into spt_server_info
  421.     values (106,"RENAME_TABLE","Y")
  422. insert into spt_server_info
  423.     values (107,"RENAME_COLUMN","Y")
  424. insert into spt_server_info
  425.     values (108,"DROP_COLUMN","N")
  426. insert into spt_server_info
  427.     values (109,"INCREASE_COLUMN_LENGTH","N")
  428. insert into spt_server_info
  429.     values (110,"DDL_IN_TRANSACTION","N")
  430. insert into spt_server_info
  431.     values (111,"DESCENDING_INDEXES","N")
  432. insert into spt_server_info
  433.     values (112,"SP_RENAME","Y")
  434. insert into spt_server_info
  435.     values (113,"REMOTE_SPROC","Y")
  436. insert into spt_server_info
  437.     values (500,"SYS_SPROC_VERSION","01.02.3015")
  438. go
  439.  
  440. grant select on spt_server_info to public
  441. go
  442.  
  443. print "creating sp_tables"
  444. go
  445.  
  446. create procedure sp_tables(
  447.                @table_name        varchar(32)  = null,
  448.                @table_owner     varchar(32)  = null,
  449.                @table_qualifier varchar(32)  = null,
  450.                @table_type        varchar(100) = null)
  451. as
  452.     declare @type1 varchar(3)
  453.     declare @tableindex int
  454.  
  455.  
  456.     /* Special feature #1:    enumerate databases when owner and name
  457.          are blank but qualifier is explicitly '%'.  */
  458.     if @table_qualifier = '%' and
  459.         @table_owner = '' and
  460.         @table_name = ''
  461.     begin    /* If enumerating databases */
  462.         select
  463.             table_qualifier = name,
  464.             table_owner = null,
  465.             table_name = null,
  466.             table_type = 'Database',
  467.             remarks = convert(varchar(254),null)    /* Remarks are NULL */
  468.         from master..sysdatabases
  469.         where name != 'model'    /* eliminate MODEL database */
  470.         order by table_qualifier
  471.     end
  472.  
  473.     /* Special feature #2:    enumerate owners when qualifier and name
  474.          are blank but owner is explicitly '%'.  */
  475.     else if @table_qualifier = '' and
  476.         @table_owner = '%' and
  477.         @table_name = ''
  478.     begin    /* If enumerating owners */
  479.         select distinct
  480.             table_qualifier = null,
  481.             table_owner = user_name(uid),
  482.             table_name = null,
  483.             table_type = 'Owner',
  484.             remarks = convert(varchar(254),null)    /* Remarks are NULL */
  485.         from sysobjects
  486.         order by table_owner
  487.     end
  488.  
  489.     else
  490.     begin /* end of special features -- do normal processing */
  491.         if @table_qualifier is not null
  492.         begin
  493.             if db_name() != @table_qualifier
  494.             begin
  495.                 if @table_qualifier = ''
  496.                 begin  /* If empty qualifier supplied */
  497.                     /* Force an empty result set */
  498.                     select @table_name = ''
  499.                     select @table_owner = ''
  500.                 end
  501.                 else
  502.                 begin    /* If qualifier doesn't match current database */
  503.                     print "Table qualifier must be name of current database"
  504.                     return
  505.                 end
  506.             end
  507.         end
  508.         if @table_type is null
  509.         begin    /* Select all ODBC supported table types */
  510.             select @type1 = 'SUV'
  511.         end
  512.         else
  513.         begin
  514.             /*    TableType is case sensitive if CS server */
  515.             select @type1 = null
  516.             if (charindex("'SYSTEM TABLE'",@table_type) != 0)
  517.                 select @type1 = @type1 + 'S'    /* Add System Tables */
  518.             if (charindex("'TABLE'",@table_type) != 0)
  519.                 select @type1 = @type1 + 'U'    /* Add User Tables */
  520.             if (charindex("'VIEW'",@table_type) != 0)
  521.                 select @type1 = @type1 + 'V'    /* Add Views */
  522.         end
  523.         if @table_name is null
  524.         begin    /*    If table name not supplied, match all */
  525.             select @table_name = '%'
  526.         end
  527.         else
  528.         begin
  529.             if (@table_owner is null) and (charindex('%', @table_name) = 0)
  530.             begin    /* If owner not specified and table is specified */
  531.                 if exists (select * from sysobjects
  532.                     where uid = user_id()
  533.                     and name = @table_name
  534.                     and (type = 'U' or type = 'V' or type = 'S'))
  535.                 begin    /* Override supplied owner w/owner of table */
  536.                     select @table_owner = user_name()
  537.                 end
  538.             end
  539.         end
  540.         if @table_owner is null /* If no owner supplied, force wildcard */
  541.             select @table_owner = '%'
  542.         select
  543.             table_qualifier = db_name(),
  544.             table_owner = user_name(o.uid),
  545.             table_name = o.name,
  546.             table_type = rtrim(
  547.                 substring('SYSTEM TABLE            TABLE       VIEW       ',
  548.                     (ascii(o.type)-83)*12+1,12)),    /* 'S'=0,'U'=2,'V'=3 */
  549.             remarks = convert(varchar(254),null)    /* Remarks are NULL */
  550.         from sysusers u, sysobjects o
  551.         where
  552.             o.name like @table_name
  553.             and user_name(o.uid) like @table_owner
  554.             and charindex(substring(o.type,1,1),@type1)! = 0 /* Only desired types */
  555.             and u.uid = user_id() /* constrain sysusers uid for use in subquery */
  556.             and (
  557.                 suser_id() = 1     /* User is the System Administrator */
  558.                 or o.uid = user_id()     /* User created the object */
  559.                 /* here's the magic... select the highest precedence of permissions in the order (user,group,public)  */
  560.                 or ((select max(((sign(uid)*abs(uid-16383))*2)+(protecttype&1))
  561.                     from sysprotects p
  562.                     /* outer join to correlate with all rows in sysobjects */
  563.                     where p.id =* o.id
  564.                         /* get rows for public,current user,user's group */
  565.                         and (p.uid = 0 or p.uid = user_id() or p.uid =* u.gid)
  566.                         /* check for SELECT,EXECUTE privilege */
  567.                         and (action in (193,224)))&1     /* more magic...normalize GRANT */
  568.                     ) = 1    /* final magic...compare Grants      */
  569.             )
  570.         order by table_type, table_qualifier, table_owner, table_name
  571.     end
  572. go
  573.  
  574. grant execute on sp_tables to public
  575. go
  576.  
  577. dump tran master with truncate_only
  578. go
  579.  
  580. print "creating sp_statistics"
  581. go
  582.  
  583. /*    Check if Version 4.8 or 4.9 */
  584. if  (charindex('4.8', @@version) = 0
  585.     and charindex('4.9', @@version) = 0)
  586. begin
  587.     print ""
  588.     print ""
  589.     print "Warning:"
  590.     print "you are installing the stored procedures "
  591.     print "on SQL Server with version less than 4.8, "
  592.     print "ignore the following error"
  593. end
  594. go
  595.  
  596. /*    This procedure is for 4.8 and 4.9 servers (catalog difference) */
  597. CREATE PROCEDURE sp_statistics (
  598.                  @table_name        varchar(32),
  599.                  @table_owner        varchar(32) = null,
  600.                  @table_qualifier    varchar(32) = null,
  601.                  @index_name        varchar(32) = '%',
  602.                  @is_unique         char(1) = 'N')
  603. AS
  604.     DECLARE @indid                int
  605.     DECLARE @lastindid            int
  606.     DECLARE @table_id            int
  607.     DECLARE @full_table_name    char(70)
  608.  
  609.     if @table_qualifier is not null
  610.     begin
  611.         if db_name() != @table_qualifier
  612.         begin    /* If qualifier doesn't match current database */
  613.             print "Table qualifier must be name of current database"
  614.             return
  615.         end
  616.     end
  617.     if @@trancount != 0
  618.     begin    /* If inside a transaction */
  619.         raiserror 20001 "stored procedure sp_statistics can not be run while in a transaction"
  620.         return
  621.     end
  622.     create table #TmpIndex(
  623.                 table_qualifier varchar(32),
  624.                 table_owner     varchar(32),
  625.                 table_name      varchar(32),
  626.                 index_qualifier varchar(32) null,
  627.                 index_name      varchar(32) null,
  628.                 non_unique      smallint null,
  629.                 type            smallint,
  630.                 seq_in_index    smallint null,
  631.                 column_name     varchar(32) null,
  632.                 collation       char(1) null,
  633.                 index_id        int null,
  634.                 cardinality     int null,
  635.                 pages           int null,
  636.                 status          smallint)
  637.     if @table_owner is null
  638.     begin    /* If unqualified table name */
  639.         SELECT @full_table_name = @table_name
  640.     end
  641.     else
  642.     begin    /* Qualified table name */
  643.         SELECT @full_table_name = @table_owner + '.' + @table_name
  644.     end
  645.     /*    Get Object ID */
  646.     SELECT @table_id = object_id(@full_table_name)
  647.  
  648.     /*    Start at lowest index id */
  649.     SELECT @indid = min(indid)
  650.     FROM sysindexes
  651.     WHERE id = @table_id
  652.         AND indid > 0
  653.         AND indid < 255
  654.  
  655.     WHILE @indid != NULL
  656.     BEGIN
  657.         INSERT #TmpIndex    /* Add all columns that are in index */
  658.             SELECT
  659.                 DB_NAME(),                                /* table_qualifier */
  660.                 USER_NAME(o.uid),                        /* table_owner       */
  661.                 o.name,                                 /* table_name       */
  662.                 o.name,                                 /* index_qualifier */
  663.                 x.name,                                 /* index_name       */
  664.                 0,                                        /* non_unique       */
  665.                 1,                                        /* SQL_INDEX_CLUSTERED */
  666.                 colid,                                    /* seq_in_index    */
  667.                 INDEX_COL(@full_table_name,indid,colid),/* column_name       */
  668.                 "A",                                    /* collation       */
  669.                 @indid,                                 /* index_id        */
  670.                 rowcnt(x.doampg),                        /* cardinality       */
  671.                 data_pgs(x.id,doampg),                    /* pages           */
  672.                 x.status                                /* status           */
  673.             FROM sysindexes x, syscolumns c, sysobjects o
  674.             WHERE
  675.                 x.id = @table_id
  676.                 AND x.id = o.id
  677.                 AND x.id = c.id
  678.                 AND c.colid < keycnt+(x.status&16)/16
  679.                 AND x.indid = @indid
  680.         /*
  681.         **      Now move @indid to the next index.
  682.         */
  683.         SELECT @lastindid = @indid
  684.         SELECT @indid = NULL
  685.  
  686.         SELECT @indid = min(indid)
  687.         FROM sysindexes
  688.         WHERE id = @table_id
  689.             AND indid > @lastindid
  690.             AND indid < 255
  691.     END
  692.  
  693.     UPDATE #TmpIndex
  694.         SET non_unique = 1
  695.         WHERE status&2 != 2 /* If non-unique index */
  696.     UPDATE #TmpIndex
  697.         SET
  698.             type = 3,            /* SQL_INDEX_OTHER */
  699.             cardinality = NULL,
  700.             pages = NULL
  701.         WHERE index_id > 1    /* If non-clustered index */
  702.  
  703.     /* now add row for table statistics */
  704.     INSERT #TmpIndex
  705.         SELECT
  706.             DB_NAME(),                /* table_qualifier */
  707.             USER_NAME(o.uid),        /* table_owner       */
  708.             o.name,                 /* table_name       */
  709.             null,                    /* index_qualifier */
  710.             null,                    /* index_name       */
  711.             null,                    /* non_unique       */
  712.             0,                        /* SQL_TABLE_STAT  */
  713.             null,                    /* seq_in_index    */
  714.             null,                    /* column_name       */
  715.             null,                    /* collation       */
  716.             0,                        /* index_id        */
  717.             rowcnt(x.doampg),        /* cardinality       */
  718.             data_pgs(x.id,doampg),    /* pages           */
  719.             0                        /* status           */
  720.         FROM sysindexes x, sysobjects o
  721.         WHERE o.id = @table_id
  722.             AND x.id = o.id
  723.             AND (x.indid = 0 or x.indid = 1)    /*    If there are no indexes */
  724.                                                 /*    then table stats are in */
  725.                                                 /*    a row with indid =0        */
  726.  
  727.     if @is_unique != 'Y'    /* If all indexes desired */
  728.         SELECT
  729.             table_qualifier,
  730.             table_owner,
  731.             table_name,
  732.             non_unique,
  733.             index_qualifier,
  734.             index_name,
  735.             type,
  736.             seq_in_index,
  737.             column_name,
  738.             collation,
  739.             cardinality,
  740.             pages
  741.         FROM #TmpIndex
  742.         WHERE
  743.             index_name like @index_name /* If matching name */
  744.             or index_name is null        /* If SQL_TABLE_STAT row */
  745.         ORDER BY non_unique, type, index_name, seq_in_index
  746.     else                    /* If only unique indexes desired */
  747.         SELECT
  748.             table_qualifier,
  749.             table_owner,
  750.             table_name,
  751.             non_unique,
  752.             index_qualifier,
  753.             index_name,
  754.             type,
  755.             seq_in_index,
  756.             column_name,
  757.             collation,
  758.             cardinality,
  759.             pages
  760.         FROM #TmpIndex
  761.         WHERE
  762.             (non_unique = 0             /* If unique */
  763.                 or non_unique is NULL)    /* If SQL_TABLE_STAT row */
  764.             and (index_name like @index_name    /* If matching name */
  765.                 or index_name is null)    /* If SQL_TABLE_STAT row */
  766.         ORDER BY non_unique, type, index_name, seq_in_index
  767.  
  768.     DROP TABLE #TmpIndex
  769. go
  770.  
  771. if object_id('#TmpIndex') != null
  772.     drop table #TmpIndex
  773. dump tran master with truncate_only
  774. go
  775.  
  776.  
  777. if  (charindex('4.8', @@version) != 0
  778.     or charindex('4.9', @@version) != 0)
  779. begin
  780.     print ""
  781.     print ""
  782.     print "Warning:"
  783.     print "you are installing the stored procedures "
  784.     print "on SQL Server with version 4.8 or greater. "
  785.     print "Ignore the following error."
  786. end
  787. go
  788.  
  789. /*    This procedure is for < 4.8 servers */
  790. CREATE PROCEDURE sp_statistics (
  791.                  @table_name        varchar(32),
  792.                  @table_owner        varchar(32) = null,
  793.                  @table_qualifier    varchar(32) = null,
  794.                  @index_name        varchar(32) = '%',
  795.                  @is_unique         char(1) = 'N')
  796. AS
  797.     DECLARE @indid                int
  798.     DECLARE @lastindid            int
  799.     DECLARE @table_id            int
  800.     DECLARE @full_table_name    char(70)
  801.  
  802.     if @table_qualifier is not null
  803.     begin
  804.         if db_name() != @table_qualifier
  805.         begin    /* If qualifier doesn't match current database */
  806.             print "Table qualifier must be name of current database"
  807.             return
  808.         end
  809.     end
  810.     if @@trancount != 0
  811.     begin    /* If inside a transaction */
  812.         raiserror 20001 "stored procedure sp_statistics can not be run while in a transaction"
  813.         return
  814.     end
  815.     create table #TmpIndex(
  816.                 table_qualifier varchar(32),
  817.                 table_owner     varchar(32),
  818.                 table_name      varchar(32),
  819.                 index_qualifier varchar(32) null,
  820.                 index_name      varchar(32) null,
  821.                 non_unique      smallint null,
  822.                 type            smallint,
  823.                 seq_in_index    smallint null,
  824.                 column_name     varchar(32) null,
  825.                 collation       char(1) null,
  826.                 index_id        int null,
  827.                 cardinality     int null,
  828.                 pages           int null,
  829.                 status          smallint)
  830.     if @table_owner is null
  831.     begin    /* If unqualified table name */
  832.         SELECT @full_table_name = @table_name
  833.     end
  834.     else
  835.     begin    /* Qualified table name */
  836.         SELECT @full_table_name = @table_owner + '.' + @table_name
  837.     end
  838.     /*    Get Object ID */
  839.     SELECT @table_id = object_id(@full_table_name)
  840.  
  841.     /*    Start at lowest index id */
  842.     SELECT @indid = min(indid)
  843.     FROM sysindexes
  844.     WHERE id = @table_id
  845.         AND indid > 0
  846.         AND indid < 255
  847.  
  848.     WHILE @indid != NULL
  849.     BEGIN
  850.         INSERT #TmpIndex    /* Add all columns that are in index */
  851.             SELECT
  852.                 DB_NAME(),                                /* table_qualifier */
  853.                 USER_NAME(o.uid),                        /* table_owner       */
  854.                 o.name,                                 /* table_name       */
  855.                 o.name,                                 /* index_qualifier */
  856.                 x.name,                                 /* index_name       */
  857.                 0,                                        /* non_unique       */
  858.                 1,                                        /* SQL_INDEX_CLUSTERED */
  859.                 colid,                                    /* seq_in_index    */
  860.                 INDEX_COL(@full_table_name,indid,colid),/* column_name       */
  861.                 "A",                                    /* collation       */
  862.                 @indid,                                 /* index_id        */
  863.                 x.rows,                                 /* cardinality       */
  864.                 x.dpages,                                /* pages           */
  865.                 x.status                                /* status            */
  866.             FROM sysindexes x, syscolumns c, sysobjects o
  867.             WHERE
  868.                 x.id = @table_id
  869.                 AND x.id = o.id
  870.                 AND x.id = c.id
  871.                 AND c.colid < keycnt+(x.status&16)/16
  872.                 AND x.indid = @indid
  873.         /*
  874.         **      Now move @indid to the next index.
  875.         */
  876.         SELECT @lastindid = @indid
  877.         SELECT @indid = NULL
  878.  
  879.         SELECT @indid = min(indid)
  880.         FROM sysindexes
  881.         WHERE id = @table_id
  882.             AND indid > @lastindid
  883.             AND indid < 255
  884.     END
  885.  
  886.     UPDATE #TmpIndex
  887.         SET non_unique = 1
  888.         WHERE status&2 != 2 /* If non-unique index */
  889.     UPDATE #TmpIndex
  890.         SET
  891.             type = 3,            /* SQL_INDEX_OTHER */
  892.             cardinality = NULL,
  893.             pages = NULL
  894.         WHERE index_id > 1    /* If non-clustered index */
  895.  
  896.     /* now add row for table statistics */
  897.     INSERT #TmpIndex
  898.         SELECT
  899.             DB_NAME(),                /* table_qualifier */
  900.             USER_NAME(o.uid),        /* table_owner       */
  901.             o.name,                 /* table_name       */
  902.             null,                    /* index_qualifier */
  903.             null,                    /* index_name       */
  904.             null,                    /* non_unique       */
  905.             0,                        /* SQL_TABLE_STAT  */
  906.             null,                    /* seq_in_index    */
  907.             null,                    /* column_name       */
  908.             null,                    /* collation       */
  909.             0,                        /* index_id        */
  910.             x.rows,                 /* cardinality       */
  911.             x.dpages,                /* pages           */
  912.             0                        /* status           */
  913.         FROM sysindexes x, sysobjects o
  914.         WHERE o.id = @table_id
  915.             AND x.id = o.id
  916.             AND (x.indid = 0 or x.indid = 1)    /*    If there are no indexes */
  917.                                                 /*    then table stats are in */
  918.                                                 /*    a row with indid =0        */
  919.  
  920.     if @is_unique != 'Y'    /* If all indexes desired */
  921.         SELECT
  922.             table_qualifier,
  923.             table_owner,
  924.             table_name,
  925.             non_unique,
  926.             index_qualifier,
  927.             index_name,
  928.             type,
  929.             seq_in_index,
  930.             column_name,
  931.             collation,
  932.             cardinality,
  933.             pages
  934.         FROM #TmpIndex
  935.         WHERE
  936.             index_name like @index_name /* If matching name */
  937.             or index_name is null        /* If SQL_TABLE_STAT row */
  938.         ORDER BY non_unique, type, index_name, seq_in_index
  939.     else                    /* If only unique indexes desired */
  940.         SELECT
  941.             table_qualifier,
  942.             table_owner,
  943.             table_name,
  944.             non_unique,
  945.             index_qualifier,
  946.             index_name,
  947.             type,
  948.             seq_in_index,
  949.             column_name,
  950.             collation,
  951.             cardinality,
  952.             pages
  953.         FROM #TmpIndex
  954.         WHERE
  955.             (non_unique = 0             /* If unique */
  956.                 or non_unique is NULL)    /* If SQL_TABLE_STAT row */
  957.             and (index_name like @index_name    /* If matching name */
  958.                 or index_name is null)    /* If SQL_TABLE_STAT row */
  959.         ORDER BY non_unique, type, index_name, seq_in_index
  960.  
  961.     DROP TABLE #TmpIndex
  962. go
  963.  
  964. grant execute on sp_statistics to public
  965. go
  966.  
  967. dump tran master with truncate_only
  968. go
  969.  
  970.  
  971. print "creating sp_columns"
  972. go
  973.  
  974. /* This is the version for servers without support for UNION */
  975. CREATE PROCEDURE sp_columns (
  976.                  @table_name        varchar(32),
  977.                  @table_owner        varchar(32) = null,
  978.                  @table_qualifier    varchar(32) = null,
  979.                  @column_name        varchar(32) = null )
  980. AS
  981.     DECLARE @full_table_name    char(70)
  982.     DECLARE @table_id int
  983.  
  984.     if @column_name is null /*    If column name not supplied, match all */
  985.         select @column_name = '%'
  986.     if @table_qualifier is not null
  987.     begin
  988.         if db_name() != @table_qualifier
  989.         begin    /* If qualifier doesn't match current database */
  990.             print "Table qualifier must be name of current database"
  991.             return
  992.         end
  993.     end
  994.     if @table_name is null
  995.     begin    /*    If table name not supplied, match all */
  996.         select @table_name = '%'
  997.     end
  998.     if @table_owner is null
  999.     begin    /* If unqualified table name */
  1000.         SELECT @full_table_name = @table_name
  1001.     end
  1002.     else
  1003.     begin    /* Qualified table name */
  1004.         SELECT @full_table_name = @table_owner + '.' + @table_name
  1005.     end
  1006.  
  1007.     /*    Get Object ID */
  1008.     SELECT @table_id = object_id(@full_table_name)
  1009.     if ((charindex('%',@full_table_name) = 0) and
  1010.         (charindex('_',@full_table_name) = 0)  and
  1011.         @table_id != 0)
  1012.     begin
  1013.         /* this block is for the case where there is no pattern
  1014.              matching required for the table name */
  1015.         SELECT
  1016.             table_qualifier = DB_NAME(),
  1017.             table_owner = USER_NAME(o.uid),
  1018.             table_name = o.name,
  1019.             column_name = c.name,
  1020.             data_type = d.data_type+convert(smallint,
  1021.                         isnull(d.aux,
  1022.                         ascii(substring("666AAA@@@CB??GG",
  1023.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))
  1024.                         -60)),
  1025.             type_name = rtrim(substring(d.type_name,
  1026.                         1+isnull(d.aux,
  1027.                         ascii(substring("III<<<MMMI<<A<A",
  1028.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1029.                         1))-60), 13)),
  1030.             precision = isnull(d.data_precision, convert(int,c.length))
  1031.                         +isnull(d.aux, convert(int,
  1032.                         ascii(substring("???AAAFFFCKFOLS",
  1033.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  1034.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  1035.                         isnull(d.aux,
  1036.                         ascii(substring("AAA<BB<DDDHJSPP",
  1037.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1038.                         1))-64)),
  1039.             scale = d.numeric_scale +convert(smallint,
  1040.                         isnull(d.aux,
  1041.                         ascii(substring("<<<<<<<<<<<<<<?",
  1042.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1043.                         1))-60)),
  1044.             radix = d.numeric_radix,
  1045.             nullable =    /* set nullability from status flag */
  1046.                 convert(smallint, convert(bit, c.status&8)),
  1047.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  1048.             ss_data_type = c.type,
  1049.             c.colid
  1050.         FROM
  1051.             syscolumns c,
  1052.             sysobjects o,
  1053.             master.dbo.spt_datatype_info d,
  1054.             systypes t
  1055.         WHERE
  1056.             o.id = @table_id
  1057.             AND c.id = o.id
  1058.             AND c.type = d.ss_dtype
  1059.             AND c.usertype *= t.usertype
  1060.             AND c.name like @column_name
  1061.         ORDER BY colid
  1062.     end
  1063.     else
  1064.     begin
  1065.         /* this block is for the case where there IS pattern
  1066.              matching done on the table name */
  1067.         if @table_owner is null /* Match all occurrances of owner */
  1068.             select @table_owner = '%'
  1069.         SELECT
  1070.             table_qualifier = DB_NAME(),
  1071.             table_owner = USER_NAME(o.uid),
  1072.             table_name = o.name,
  1073.             column_name = c.name,
  1074.             data_type = d.data_type+convert(smallint,
  1075.                         isnull(d.aux,
  1076.                         ascii(substring("666AAA@@@CB??GG",
  1077.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))
  1078.                         -60)),
  1079.             type_name = rtrim(substring(d.type_name,
  1080.                         1+isnull(d.aux,
  1081.                         ascii(substring("III<<<MMMI<<A<A",
  1082.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1083.                         1))-60), 13)),
  1084.             precision = isnull(d.data_precision, convert(int,c.length))
  1085.                         +isnull(d.aux, convert(int,
  1086.                         ascii(substring("???AAAFFFCKFOLS",
  1087.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  1088.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  1089.                         isnull(d.aux,
  1090.                         ascii(substring("AAA<BB<DDDHJSPP",
  1091.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1092.                         1))-64)),
  1093.             scale = d.numeric_scale +convert(smallint,
  1094.                         isnull(d.aux,
  1095.                         ascii(substring("<<<<<<<<<<<<<<?",
  1096.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1097.                         1))-60)),
  1098.             radix = d.numeric_radix,
  1099.             nullable =    /* set nullability from status flag */
  1100.                 convert(smallint, convert(bit, c.status&8)),
  1101.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  1102.             ss_data_type = c.type,
  1103.             c.colid
  1104.         FROM
  1105.             syscolumns c,
  1106.             sysobjects o,
  1107.             master.dbo.spt_datatype_info d,
  1108.             systypes t
  1109.         WHERE
  1110.             o.name like @table_name
  1111.             AND user_name(o.uid) like @table_owner
  1112.             AND c.id = o.id
  1113.             AND c.usertype *= t.usertype
  1114.             AND c.type = d.ss_dtype
  1115.             AND o.type != 'P'
  1116.             AND c.name like @column_name
  1117.         ORDER BY table_qualifier, table_owner, table_name, colid
  1118.     end
  1119. go
  1120.  
  1121. /* if this is a 4.2 or later SQL Server, then we want to delete the stored
  1122. procedure we just created, and create a new one which uses UNION
  1123. (this fixes some bugs involving UDT names and char NULL datatype
  1124. names)
  1125. */
  1126.  
  1127. if  (charindex('1.1', @@version) = 0
  1128.     and charindex('4.0', @@version) = 0)
  1129. begin
  1130.     drop proc sp_columns
  1131.     dump tran master with truncate_only
  1132. end
  1133. else
  1134. begin
  1135.     print ""
  1136.     print ""
  1137.     print "Installing catalog procedures on a 1.x or 4.0 server:"
  1138.     print "Ignore the following error messages"
  1139.     print ""
  1140.     print ""
  1141. end
  1142. go
  1143.  
  1144. /* This is the version for servers which support UNION */
  1145. CREATE PROCEDURE sp_columns (
  1146.                  @table_name        varchar(32),
  1147.                  @table_owner        varchar(32) = null,
  1148.                  @table_qualifier    varchar(32) = null,
  1149.                  @column_name        varchar(32) = null )
  1150. AS
  1151.     DECLARE @full_table_name    char(70)
  1152.     DECLARE @table_id int
  1153.  
  1154.     if @column_name is null /*    If column name not supplied, match all */
  1155.         select @column_name = '%'
  1156.     if @table_qualifier is not null
  1157.     begin
  1158.         if db_name() != @table_qualifier
  1159.         begin    /* If qualifier doesn't match current database */
  1160.             print "Table qualifier must be name of current database"
  1161.             return
  1162.         end
  1163.     end
  1164.     if @table_name is null
  1165.     begin    /*    If table name not supplied, match all */
  1166.         select @table_name = '%'
  1167.     end
  1168.     if @table_owner is null
  1169.     begin    /* If unqualified table name */
  1170.         SELECT @full_table_name = @table_name
  1171.     end
  1172.     else
  1173.     begin    /* Qualified table name */
  1174.         SELECT @full_table_name = @table_owner + '.' + @table_name
  1175.     end
  1176.  
  1177.     /*    Get Object ID */
  1178.     SELECT @table_id = object_id(@full_table_name)
  1179.     if ((charindex('%',@full_table_name) = 0) and
  1180.         (charindex('_',@full_table_name) = 0)  and
  1181.         @table_id != 0)
  1182.     begin
  1183.         /* this block is for the case where there is no pattern
  1184.              matching required for the table name */
  1185.         SELECT    /* INTn, FLOATn, DATETIMEn and MONEYn types */
  1186.             table_qualifier = DB_NAME(),
  1187.             table_owner = USER_NAME(o.uid),
  1188.             table_name = o.name,
  1189.             column_name = c.name,
  1190.             data_type = d.data_type+convert(smallint,
  1191.                         isnull(d.aux,
  1192.                         ascii(substring("666AAA@@@CB??GG",
  1193.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))
  1194.                         -60)),
  1195.             type_name = rtrim(substring(d.type_name,
  1196.                         1+isnull(d.aux,
  1197.                         ascii(substring("III<<<MMMI<<A<A",
  1198.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1199.                         1))-60), 13)),
  1200.             precision = isnull(d.data_precision, convert(int,c.length))
  1201.                         +isnull(d.aux, convert(int,
  1202.                         ascii(substring("???AAAFFFCKFOLS",
  1203.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  1204.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  1205.                         isnull(d.aux,
  1206.                         ascii(substring("AAA<BB<DDDHJSPP",
  1207.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1208.                         1))-64)),
  1209.             scale = d.numeric_scale +convert(smallint,
  1210.                         isnull(d.aux,
  1211.                         ascii(substring("<<<<<<<<<<<<<<?",
  1212.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1213.                         1))-60)),
  1214.             radix = d.numeric_radix,
  1215.             nullable =    /* set nullability from status flag */
  1216.                 convert(smallint, convert(bit, c.status&8)),
  1217.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  1218.             ss_data_type = c.type,
  1219.             colid = c.colid
  1220.         FROM
  1221.             syscolumns c,
  1222.             sysobjects o,
  1223.             master.dbo.spt_datatype_info d,
  1224.             systypes t
  1225.         WHERE
  1226.             o.id = @table_id
  1227.             AND c.id = o.id
  1228.             AND c.type = d.ss_dtype
  1229.             AND c.name like @column_name
  1230.             AND d.ss_dtype IN (111, 109, 38, 110)    /* Just *N types */
  1231.             AND c.usertype < 100                    /* No user defined types */
  1232.         UNION
  1233.         SELECT    /* All other types including user data types */
  1234.             table_qualifier = DB_NAME(),
  1235.             table_owner = USER_NAME(o.uid),
  1236.             table_name = o.name,
  1237.             column_name = c.name,
  1238.             data_type = convert(smallint,
  1239.                 /*    Map systypes.type to ODBC type */
  1240.                 /*        SS-Type         "                 1          "*/
  1241.                 /*                        "33 3 3 4 44 5 5 2 5 55666"*/
  1242.                 /*                        "45 7 9    5 78 0 2 2 6 89012"*/
  1243.                         ascii(substring("8;<9<H<<<<<:<=6<5<A<?<@<GC?GD",
  1244.                         t.type%34+1,1))-60),
  1245.             type_name = t.name,
  1246.             precision = isnull(d.data_precision, convert(int,c.length))
  1247.                         +isnull(d.aux, convert(int,
  1248.                         ascii(substring("???AAAFFFCKFOLS",
  1249.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  1250.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  1251.                         isnull(d.aux,
  1252.                         ascii(substring("AAA<BB<DDDHJSPP",
  1253.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1254.                         1))-64)),
  1255.             scale = d.numeric_scale +convert(smallint,
  1256.                         isnull(d.aux,
  1257.                         ascii(substring("<<<<<<<<<<<<<<?",
  1258.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1259.                         1))-60)),
  1260.             radix = d.numeric_radix,
  1261.             nullable =    /* set nullability from status flag */
  1262.                 convert(smallint, convert(bit, c.status&8)),
  1263.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  1264.             ss_data_type = c.type,
  1265.             colid = c.colid
  1266.         FROM
  1267.             syscolumns c,
  1268.             sysobjects o,
  1269.             master.dbo.spt_datatype_info d,
  1270.             systypes t
  1271.         WHERE
  1272.             o.id = @table_id
  1273.             AND c.id = o.id
  1274.             AND c.type = d.ss_dtype
  1275.             AND c.usertype *= t.usertype
  1276.             AND c.name like @column_name
  1277.             AND (d.ss_dtype NOT IN (111, 109, 38, 110)    /* No *N types */
  1278.                 OR c.usertype >= 100)                    /* User defined types */
  1279.         ORDER BY colid
  1280.     end
  1281.     else
  1282.     begin
  1283.         /* this block is for the case where there IS pattern
  1284.              matching done on the table name */
  1285.         if @table_owner is null /*    If owner not supplied, match all */
  1286.             select @table_owner = '%'
  1287.         SELECT    /* INTn, FLOATn, DATETIMEn and MONEYn types */
  1288.             table_qualifier = DB_NAME(),
  1289.             table_owner = USER_NAME(o.uid),
  1290.             table_name = o.name,
  1291.             column_name = c.name,
  1292.             data_type = d.data_type+convert(smallint,
  1293.                         isnull(d.aux,
  1294.                         ascii(substring("666AAA@@@CB??GG",
  1295.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))
  1296.                         -60)),
  1297.             type_name = rtrim(substring(d.type_name,
  1298.                         1+isnull(d.aux,
  1299.                         ascii(substring("III<<<MMMI<<A<A",
  1300.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1301.                         1))-60), 13)),
  1302.             precision = isnull(d.data_precision, convert(int,c.length))
  1303.                         +isnull(d.aux, convert(int,
  1304.                         ascii(substring("???AAAFFFCKFOLS",
  1305.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  1306.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  1307.                         isnull(d.aux,
  1308.                         ascii(substring("AAA<BB<DDDHJSPP",
  1309.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1310.                         1))-64)),
  1311.             scale = d.numeric_scale +convert(smallint,
  1312.                         isnull(d.aux,
  1313.                         ascii(substring("<<<<<<<<<<<<<<?",
  1314.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1315.                         1))-60)),
  1316.             radix = d.numeric_radix,
  1317.             nullable =    /* set nullability from status flag */
  1318.                 convert(smallint, convert(bit, c.status&8)),
  1319.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  1320.             ss_data_type = c.type,
  1321.             colid = c.colid
  1322.         FROM
  1323.             syscolumns c,
  1324.             sysobjects o,
  1325.             master.dbo.spt_datatype_info d,
  1326.             systypes t
  1327.         WHERE
  1328.             o.name like @table_name
  1329.             AND user_name(o.uid) like @table_owner
  1330.             AND o.id = c.id
  1331.             AND c.type = d.ss_dtype
  1332.             AND o.type != 'P'
  1333.             AND c.name like @column_name
  1334.             AND d.ss_dtype IN (111, 109, 38, 110)    /* Just *N types */
  1335.         UNION
  1336.         SELECT    /* All other types including user data types */
  1337.             table_qualifier = DB_NAME(),
  1338.             table_owner = USER_NAME(o.uid),
  1339.             table_name = o.name,
  1340.             column_name = c.name,
  1341.             data_type = convert(smallint,
  1342.                 /*    Map systypes.type to ODBC type */
  1343.                 /*        SS-Type         "                 1          "*/
  1344.                 /*                        "33 3 3 4 44 5 5 2 5 55666"*/
  1345.                 /*                        "45 7 9    5 78 0 2 2 6 89012"*/
  1346.                         ascii(substring("8;<9<H<<<<<:<=6<5<A<?<@<GC?GD",
  1347.                         t.type%34+1,1))-60),
  1348.             type_name = t.name,
  1349.             precision = isnull(d.data_precision, convert(int,c.length))
  1350.                         +isnull(d.aux, convert(int,
  1351.                         ascii(substring("???AAAFFFCKFOLS",
  1352.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  1353.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  1354.                         isnull(d.aux,
  1355.                         ascii(substring("AAA<BB<DDDHJSPP",
  1356.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1357.                         1))-64)),
  1358.             scale = d.numeric_scale +convert(smallint,
  1359.                         isnull(d.aux,
  1360.                         ascii(substring("<<<<<<<<<<<<<<?",
  1361.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  1362.                         1))-60)),
  1363.             radix = d.numeric_radix,
  1364.             nullable =    /* set nullability from status flag */
  1365.                 convert(smallint, convert(bit, c.status&8)),
  1366.             remarks  = convert(varchar(254),null),
  1367.             ss_data_type = c.type,
  1368.             colid = c.colid
  1369.         FROM
  1370.             syscolumns c,
  1371.             sysobjects o,
  1372.             master.dbo.spt_datatype_info d,
  1373.             systypes t
  1374.         WHERE
  1375.             o.name like @table_name
  1376.             AND user_name(o.uid) like @table_owner
  1377.             AND o.id = c.id
  1378.             AND c.type = d.ss_dtype
  1379.             AND c.usertype *= t.usertype
  1380.             AND o.type != 'P'
  1381.             AND c.name like @column_name
  1382.             AND d.ss_dtype NOT IN (111, 109, 38, 110)    /* No *N types */
  1383.         ORDER BY table_owner, table_name, colid
  1384.     end
  1385. go
  1386.  
  1387. grant execute on sp_columns to public
  1388. go
  1389.  
  1390. dump tran master with truncate_only
  1391. go
  1392. print "creating sp_fkeys"
  1393. go
  1394.  
  1395. CREATE PROCEDURE sp_fkeys(
  1396.                @pktable_name        varchar(32) = null,
  1397.                @pktable_owner        varchar(32) = null,
  1398.                @pktable_qualifier    varchar(32) = null,
  1399.                @fktable_name        varchar(32) = null,
  1400.                @fktable_owner        varchar(32) = null,
  1401.                @fktable_qualifier    varchar(32) = null )
  1402. as
  1403.     declare     @order_by_pk int
  1404.  
  1405.     select  @order_by_pk = 0
  1406.  
  1407.     if (@pktable_name is null) and (@fktable_name is null)
  1408.     begin    /* If neither primary key nor foreign key table names given */
  1409.         print "pk table name or fk table name must be given"
  1410.         return
  1411.     end
  1412.     if @fktable_qualifier is not null
  1413.     begin
  1414.         if db_name() != @fktable_qualifier
  1415.         begin    /* If qualifier doesn't match current database */
  1416.             print "Foreign Key Table qualifier must be name of current database"
  1417.             return
  1418.         end
  1419.     end
  1420.     if @pktable_qualifier is not null
  1421.     begin
  1422.         if db_name() != @pktable_qualifier
  1423.         begin    /* If qualifier doesn't match current database */
  1424.             print "Primary Key Table qualifier must be name of current database"
  1425.             return
  1426.         end
  1427.     end
  1428.  
  1429.     if @pktable_name is null
  1430.     begin /*  If table name not supplied, match all */
  1431.         select @pktable_name = '%'
  1432.         select @order_by_pk = 1
  1433.     end
  1434.     if @pktable_owner is null    /*    If PK owner not supplied, match all */
  1435.         select @pktable_owner = '%'
  1436.     if @fktable_name is null    /*    If table name not supplied, match all */
  1437.         select @fktable_name = '%'
  1438.     if @fktable_owner is null    /*    If FK owner not supplied, match all */
  1439.         select @fktable_owner = '%'
  1440.  
  1441.     if @@trancount != 0
  1442.     begin    /* If inside a transaction */
  1443.         raiserror 20001 "catalog procedure sp_fkeys can not be run in a transaction"
  1444.     return
  1445.     end
  1446.     create table #fkeys(
  1447.              pktable_qualifier    varchar(32),
  1448.              pktable_owner        varchar(32),
  1449.              pktable_name       varchar(32),
  1450.              pkcolumn_name      varchar(32),
  1451.              fktable_qualifier    varchar(32),
  1452.              fktable_owner        varchar(32),
  1453.              fktable_name       varchar(32),
  1454.              fkcolumn_name        varchar(32),
  1455.              key_seq            smallint)
  1456.  
  1457.     /*    SQL Server supports upto 8 PK/FK relationships between 2 tables */
  1458.     /*    Process syskeys for each relationship */
  1459.     /*    The inserts below adds a row to the temp table for each of the
  1460.         8 possible relationships */
  1461.     insert into #fkeys
  1462.         select
  1463.             db_name(),
  1464.             (select user_name(uid) from sysobjects o where o.id = k.depid),
  1465.             object_name(k.depid),
  1466.             (select name from syscolumns
  1467.                 where id = k.depid    and colid = k.depkey1),
  1468.             db_name(),
  1469.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1470.             object_name(k.id),
  1471.             c.name,
  1472.             1
  1473.         from
  1474.             syskeys k, syscolumns c
  1475.         where
  1476.             c.id = k.id
  1477.             and k.type = 2    /* Foreign type key */
  1478.             and c.colid = key1
  1479.     if (@@rowcount = 0)
  1480.         goto done
  1481.  
  1482.     insert into #fkeys
  1483.         select
  1484.             db_name(),
  1485.             (select user_name(uid) from sysobjects o where o.id = k.depid),
  1486.             object_name(k.depid),
  1487.             (select name from syscolumns
  1488.                 where id = k.depid and colid = k.depkey2),
  1489.             db_name(),
  1490.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1491.             object_name(k.id),
  1492.             c.name,
  1493.             2
  1494.         from
  1495.             syskeys k, syscolumns c
  1496.         where
  1497.             c.id = k.id
  1498.             and k.type = 2    /* Foreign type key */
  1499.             and c.colid = key2
  1500.     if (@@rowcount = 0)
  1501.         goto done
  1502.  
  1503.     insert into #fkeys
  1504.         select
  1505.             db_name(),
  1506.             (select user_name(uid) from sysobjects o where o.id = k.depid),
  1507.             object_name(k.depid),
  1508.             (select name from syscolumns
  1509.                 where id = k.depid and colid = k.depkey3),
  1510.             db_name(),
  1511.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1512.             object_name(k.id),
  1513.             c.name,
  1514.             3
  1515.         from
  1516.             syskeys k, syscolumns c
  1517.         where
  1518.             c.id = k.id
  1519.             and k.type = 2    /* Foreign type key */
  1520.             and c.colid = key3
  1521.     if (@@rowcount = 0)
  1522.         goto done
  1523.  
  1524.     insert into #fkeys
  1525.         select
  1526.             db_name(),
  1527.             (select user_name(uid) from sysobjects o where o.id = k.depid),
  1528.             object_name(k.depid),
  1529.             (select name from syscolumns
  1530.                 where id = k.depid and colid = k.depkey4),
  1531.             db_name(),
  1532.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1533.             object_name(k.id),
  1534.             c.name,
  1535.             4
  1536.         from
  1537.             syskeys k, syscolumns c
  1538.         where
  1539.             c.id = k.id
  1540.             and k.type = 2    /* Foreign type key */
  1541.             and c.colid = key4
  1542.     if (@@rowcount = 0)
  1543.         goto done
  1544.  
  1545.     insert into #fkeys
  1546.         select
  1547.             db_name(),
  1548.             (select user_name(uid) from sysobjects o where o.id = k.depid),
  1549.             object_name(k.depid),
  1550.             (select name from syscolumns
  1551.                 where id = k.depid and colid = k.depkey5),
  1552.             db_name(),
  1553.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1554.             object_name(k.id),
  1555.             c.name,
  1556.             5
  1557.         from
  1558.             syskeys k, syscolumns c
  1559.         where
  1560.             c.id = k.id
  1561.             and k.type = 2    /* Foreign type key */
  1562.             and c.colid = key5
  1563.  
  1564.     if (@@rowcount = 0)
  1565.         goto done
  1566.  
  1567.     insert into #fkeys
  1568.         select
  1569.             db_name(),
  1570.             (select user_name(uid) from sysobjects o where o.id = k.depid),
  1571.             object_name(k.depid),
  1572.             (select name from syscolumns
  1573.                 where id = k.depid and colid = k.depkey6),
  1574.             db_name(),
  1575.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1576.             object_name(k.id),
  1577.             c.name,
  1578.             6
  1579.         from
  1580.             syskeys k, syscolumns c
  1581.         where
  1582.             c.id = k.id
  1583.             and k.type = 2    /* Foreign type key */
  1584.             and c.colid = key6
  1585.     if (@@rowcount = 0)
  1586.         goto done
  1587.  
  1588.     insert into #fkeys
  1589.         select
  1590.             db_name(),
  1591.             (select user_name(uid) from sysobjects o where o.id = k.depid),
  1592.             object_name(k.depid),
  1593.             (select name from syscolumns
  1594.                 where id = k.depid and colid = k.depkey7),
  1595.             db_name(),
  1596.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1597.             object_name(k.id),
  1598.             c.name,
  1599.             7
  1600.         from
  1601.             syskeys k, syscolumns c
  1602.         where
  1603.             c.id = k.id
  1604.             and k.type = 2    /* Foreign type key */
  1605.             and c.colid = key7
  1606.     if (@@rowcount = 0)
  1607.         goto done
  1608.  
  1609.     insert into #fkeys
  1610.         select
  1611.             db_name(),
  1612.             (select user_name(uid) from sysobjects o where o.id = k.depid),
  1613.             object_name(k.depid),
  1614.             (select name from syscolumns
  1615.                 where id = k.depid and colid = k.depkey8),
  1616.             db_name(),
  1617.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1618.             object_name(k.id),
  1619.             c.name,
  1620.             8
  1621.         from
  1622.             syskeys k, syscolumns c
  1623.         where
  1624.             c.id = k.id
  1625.             and k.type = 2    /* Foreign type key */
  1626.             and c.colid = key8
  1627.     done:
  1628.     if @order_by_pk = 1 /*    If order by PK fields */
  1629.         select
  1630.             pktable_qualifier,
  1631.             pktable_owner,
  1632.             pktable_name,
  1633.             pkcolumn_name,
  1634.             fktable_qualifier,
  1635.             fktable_owner,
  1636.             fktable_name,
  1637.             fkcolumn_name,
  1638.             key_seq,
  1639.             update_rule = convert(smallint, null),
  1640.             delete_rule = convert(smallint,null)
  1641.         from #fkeys
  1642.         where fktable_name like @fktable_name
  1643.             and fktable_owner like @fktable_owner
  1644.             and pktable_name  like @pktable_name
  1645.             and pktable_owner like @pktable_owner
  1646.         order by pktable_qualifier, pktable_owner, pktable_name, key_seq
  1647.     else        /*    Order by FK fields */
  1648.         select
  1649.             pktable_qualifier,
  1650.             pktable_owner,
  1651.             pktable_name,
  1652.             pkcolumn_name,
  1653.             fktable_qualifier,
  1654.             fktable_owner,
  1655.             fktable_name,
  1656.             fkcolumn_name,
  1657.             key_seq,
  1658.             update_rule = convert(smallint,null),
  1659.             delete_rule = convert(smallint,null)
  1660.         from #fkeys
  1661.         where fktable_name like @fktable_name
  1662.             and fktable_owner like @fktable_owner
  1663.             and pktable_name  like @pktable_name
  1664.             and pktable_owner like @pktable_owner
  1665.         order by fktable_qualifier, fktable_owner, fktable_name, key_seq
  1666. go
  1667.  
  1668. grant execute on sp_fkeys to public
  1669. go
  1670. dump tran master with truncate_only
  1671. go
  1672.  
  1673. print "creating sp_pkeys"
  1674. go
  1675.  
  1676. CREATE PROCEDURE sp_pkeys(
  1677.                @table_name        varchar(32),
  1678.                @table_owner     varchar(32) = null,
  1679.                @table_qualifier varchar(32) = null )
  1680. as
  1681.     if @table_qualifier is not null
  1682.     begin
  1683.         if db_name() != @table_qualifier
  1684.         begin    /* If qualifier doesn't match current database */
  1685.             print "Table qualifier must be name of current database"
  1686.             return
  1687.         end
  1688.     end
  1689.     if @table_owner is null /*    If owner not supplied, match all */
  1690.         select @table_owner = '%'
  1691.     if @@trancount != 0
  1692.     begin    /* If inside a transaction */
  1693.         raiserror 20001 "catalog procedure sp_pkeys can not be run in a transaction"
  1694.         return
  1695.     end
  1696.  
  1697.     create table #pkeys(
  1698.              table_qualifier varchar(32),
  1699.              table_owner     varchar(32),
  1700.              table_name      varchar(32),
  1701.              column_name     varchar(32),
  1702.              key_seq         smallint)
  1703.  
  1704.     /*    SQL Server supports upto 8 PK/FK relationships between 2 tables */
  1705.     /*    Process syskeys for each relationship */
  1706.     /*    The inserts below adds a row to the temp table for each of the
  1707.         8 possible relationships */
  1708.     insert into #pkeys
  1709.         select
  1710.             db_name(),
  1711.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1712.             object_name(k.id),
  1713.             c.name,
  1714.             1
  1715.         from
  1716.             syskeys k, syscolumns c
  1717.         where
  1718.             c.id = k.id
  1719.             and k.type = 1    /* Primary type key */
  1720.             and c.colid = k.key1
  1721.     if (@@rowcount = 0)
  1722.         goto done
  1723.  
  1724.     insert into #pkeys
  1725.         select
  1726.             db_name(),
  1727.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1728.             object_name(k.id),
  1729.             c.name,
  1730.             2
  1731.         from
  1732.             syskeys k, syscolumns c
  1733.         where
  1734.             c.id = k.id
  1735.             and k.type = 1    /* Primary type key */
  1736.             and c.colid = key2
  1737.     if (@@rowcount = 0)
  1738.         goto done
  1739.  
  1740.     insert into #pkeys
  1741.         select
  1742.             db_name(),
  1743.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1744.             object_name(k.id),
  1745.             c.name,
  1746.             3
  1747.         from
  1748.             syskeys k, syscolumns c
  1749.         where
  1750.             c.id = k.id
  1751.             and k.type = 1    /* Primary type key */
  1752.             and c.colid = key3
  1753.     if (@@rowcount = 0)
  1754.         goto done
  1755.  
  1756.     insert into #pkeys
  1757.         select
  1758.             db_name(),
  1759.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1760.             object_name(k.id),
  1761.             c.name,
  1762.             4
  1763.         from
  1764.             syskeys k, syscolumns c
  1765.         where
  1766.             c.id = k.id
  1767.             and k.type = 1    /* Primary type key */
  1768.             and c.colid = key4
  1769.     if (@@rowcount = 0)
  1770.         goto done
  1771.  
  1772.     insert into #pkeys
  1773.         select
  1774.             db_name(),
  1775.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1776.             object_name(k.id),
  1777.             c.name,
  1778.             5
  1779.         from
  1780.             syskeys k, syscolumns c
  1781.         where
  1782.             c.id = k.id
  1783.             and k.type = 1    /* Primary type key */
  1784.             and c.colid = key5
  1785.     if (@@rowcount = 0)
  1786.         goto done
  1787.  
  1788.     insert into #pkeys
  1789.         select
  1790.             db_name(),
  1791.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1792.             object_name(k.id),
  1793.             c.name,
  1794.             6
  1795.         from
  1796.             syskeys k, syscolumns c
  1797.         where
  1798.             c.id = k.id
  1799.             and k.type = 1    /* Primary type key */
  1800.             and c.colid = key6
  1801.     if (@@rowcount = 0)
  1802.         goto done
  1803.  
  1804.     insert into #pkeys
  1805.         select
  1806.             db_name(),
  1807.             (select user_name(uid) from sysobjects o where o.id = k.id),
  1808.             object_name(k.id),
  1809.             c.name,
  1810.             7
  1811.         from
  1812.             syskeys k, syscolumns c
  1813.         where
  1814.             c.id = k.id
  1815.             and k.type = 1    /* Primary type key */
  1816.             and c.colid = key7
  1817.     if (@@rowcount = 0)
  1818.         goto done
  1819.  
  1820.     insert into #pkeys
  1821.          select
  1822.              db_name(),
  1823.              (select user_name(uid) from sysobjects o where o.id = k.id),
  1824.              object_name(k.id),
  1825.              c.name,
  1826.              8
  1827.          from
  1828.              syskeys k, syscolumns c
  1829.          where
  1830.              c.id = k.id
  1831.              and k.type = 1 /* Primary type key */
  1832.              and c.colid = key8
  1833.  
  1834.     done:
  1835.     select
  1836.         table_qualifier,
  1837.         table_owner,
  1838.         table_name,
  1839.         column_name,
  1840.         key_seq
  1841.         from #pkeys
  1842.     where table_name like  @table_name
  1843.         and table_owner like @table_owner
  1844.     order by table_qualifier, table_owner, table_name, key_seq
  1845. go
  1846.  
  1847. grant execute on sp_pkeys to public
  1848. go
  1849.  
  1850. dump tran master with truncate_only
  1851. go
  1852.  
  1853. print "creating sp_stored_procedures"
  1854. go
  1855.  
  1856. create procedure sp_stored_procedures(
  1857.                         @sp_name        varchar(36) = null,
  1858.                         @sp_owner        varchar(32) = null,
  1859.                         @sp_qualifier    varchar(32) = null)
  1860. as
  1861.     if @sp_qualifier is not null
  1862.     begin
  1863.         if db_name() != @sp_qualifier
  1864.         begin
  1865.             if @sp_qualifier = ''
  1866.             begin
  1867.                 /* in this case, we need to return an empty result set */
  1868.                 /* because the user has requested a database with an empty name */
  1869.                 select @sp_name = ''
  1870.                 select @sp_owner = ''
  1871.             end else
  1872.             begin    /* If qualifier doesn't match current database */
  1873.                 print "Stored procedure qualifier must be name of current database"
  1874.                 return
  1875.             end
  1876.         end
  1877.     end
  1878.  
  1879.     if @sp_name is null
  1880.     begin  /*  If procedure name not supplied, match all */
  1881.         select @sp_name = '%'
  1882.     end
  1883.     else begin
  1884.         if (@sp_owner is null) and (charindex('%', @sp_name) = 0)
  1885.         begin
  1886.             if exists (select * from sysobjects
  1887.                 where uid = user_id()
  1888.                     and name = @sp_name
  1889.                     and type = 'P') /* Object type of Procedure */
  1890.             begin
  1891.                 select @sp_owner = user_name()
  1892.             end
  1893.         end
  1894.     end
  1895.     if @sp_owner is null    /*    If procedure owner not supplied, match all */
  1896.         select @sp_owner = '%'
  1897.  
  1898.     select
  1899.         procedure_qualifier = db_name(),
  1900.         procedure_owner = user_name(o.uid),
  1901.         procedure_name = o.name +';'+ ltrim(str(p.number,5)),
  1902.         num_input_params = -1,    /* Constant since value unknown */
  1903.         num_output_params = -1, /* Constant since value unknown */
  1904.         num_result_sets = -1,    /* Constant since value unknown */
  1905.         remarks = convert(varchar(254),null)    /* Remarks are NULL */
  1906.     from
  1907.         sysobjects o,sysprocedures p,sysusers u
  1908.     where
  1909.         o.name like @sp_name
  1910.         and p.sequence = 0
  1911.         and user_name(o.uid) like @sp_owner
  1912.         and o.type = 'P'        /* Object type of Procedure */
  1913.         and p.id = o.id
  1914.         and u.uid = user_id()    /* constrain sysusers uid for use in subquery */
  1915.         and (suser_id() = 1     /* User is the System Administrator */
  1916.             or o.uid = user_id()    /* User created the object */
  1917.             /* here's the magic... select the highest precedence of permissions in the order (user,group,public)  */
  1918.             or ((select max(((sign(uid)*abs(uid-16383))*2)+(protecttype&1))
  1919.              from sysprotects p
  1920.              /* outer join to correlate with all rows in sysobjects */
  1921.              where p.id =* o.id
  1922.                  /*  get rows for public,current user,user's group */
  1923.                  and (p.uid = 0 or p.uid = user_id() or p.uid =* u.gid)
  1924.                  /* check for SELECT,EXECUTE privilege */
  1925.                  and (action in (193,224)))&1    /* more magic...normalize GRANT */
  1926.             ) = 1     /* final magic...compare Grants    */
  1927.         )
  1928.     order by procedure_qualifier, procedure_owner, procedure_name
  1929. go
  1930. grant execute on sp_stored_procedures to public
  1931. go
  1932.  
  1933. dump tran master with truncate_only
  1934. go
  1935.  
  1936.  
  1937. print "creating sp_sproc_columns"
  1938. go
  1939.  
  1940. /* this is the version for servers  without support for UNION */
  1941. CREATE PROCEDURE sp_sproc_columns (
  1942.                  @procedure_name        varchar(36) = '%',
  1943.                  @procedure_owner        varchar(32) = null,
  1944.                  @procedure_qualifier    varchar(32) = null,
  1945.                  @column_name            varchar(32) = null )
  1946. AS
  1947.  
  1948.     DECLARE @group_num int
  1949.     DECLARE @semi_position int
  1950.     DECLARE @full_procedure_name    char(70)
  1951.     DECLARE @procedure_id int
  1952.  
  1953.     if @column_name is null /*    If column name not supplied, match all */
  1954.         select @column_name = '%'
  1955.     if @procedure_qualifier is not null
  1956.     begin
  1957.         if db_name() != @procedure_qualifier
  1958.         begin
  1959.             if @procedure_qualifier = ''
  1960.             begin
  1961.                 /* in this case, we need to return an empty result set */
  1962.                 /* because the user has requested a database with an empty name */
  1963.                 select @procedure_name = ''
  1964.                 select @procedure_owner = ''
  1965.             end else
  1966.             begin    /* If qualifier doesn't match current database */
  1967.                 print "Procedure qualifier must be name of current database"
  1968.                 return
  1969.             end
  1970.         end
  1971.     end
  1972.     if @procedure_name is null
  1973.     begin    /*    If procedure name not supplied, match all */
  1974.         select @procedure_name = '%'
  1975.     end
  1976.     /* first we need to extract the procedure group number, if one exists */
  1977.     select @semi_position = charindex(';',@procedure_name)
  1978.     if (@semi_position > 0)
  1979.     begin    /* If group number separator (;) found */
  1980.         select @group_num =
  1981.             convert(int,substring(@procedure_name, @semi_position + 1, 2))
  1982.         select @procedure_name =
  1983.             substring(@procedure_name, 1, @semi_position -1)
  1984.     end
  1985.     else
  1986.     begin    /* No group separator, so default to group number of 1 */
  1987.         select @group_num = 1
  1988.     end
  1989.     if @procedure_owner is null
  1990.     begin    /* No proc owner, so default to current user */
  1991.         SELECT @full_procedure_name = @procedure_name
  1992.     end
  1993.     else
  1994.     begin    /* Build qualified name using proc owner */
  1995.         SELECT @full_procedure_name = @procedure_owner + '.' + @procedure_name
  1996.     end
  1997.  
  1998.     /*    Get Object ID */
  1999.     SELECT @procedure_id = object_id(@full_procedure_name)
  2000.     if ((charindex('%',@full_procedure_name) = 0) and
  2001.         (charindex('_',@full_procedure_name) = 0)  and
  2002.         @procedure_id != 0)
  2003.     begin
  2004.     /* this block is for the case where there is no pattern
  2005.         matching required for the procedure name */
  2006.         SELECT
  2007.             procedure_qualifier = DB_NAME(),
  2008.             procedure_owner = USER_NAME(o.uid),
  2009.             procedure_name = o.name +';'+ ltrim(str(c.number,5)),
  2010.             column_name = c.name,
  2011.             column_type = convert(smallint, 0),
  2012.             data_type = d.data_type+convert(smallint,
  2013.                         isnull(d.aux,
  2014.                         ascii(substring("666AAA@@@CB??GG",
  2015.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))
  2016.                         -60)),
  2017.             type_name = rtrim(substring(d.type_name,
  2018.                         1+isnull(d.aux,
  2019.                         ascii(substring("III<<<MMMI<<A<A",
  2020.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2021.                         1))-60), 13)),
  2022.             precision = isnull(d.data_precision, convert(int,c.length))
  2023.                         +isnull(d.aux, convert(int,
  2024.                         ascii(substring("???AAAFFFCKFOLS",
  2025.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  2026.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  2027.                         isnull(d.aux,
  2028.                         ascii(substring("AAA<BB<DDDHJSPP",
  2029.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2030.                         1))-64)),
  2031.             scale = d.numeric_scale +convert(smallint,
  2032.                         isnull(d.aux,
  2033.                         ascii(substring("<<<<<<<<<<<<<<?",
  2034.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2035.                         1))-60)),
  2036.             radix = d.numeric_radix,
  2037.             nullable = convert(smallint,t.allownulls),
  2038.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  2039.             ss_data_type = c.type
  2040.         FROM
  2041.             syscolumns c,
  2042.             sysobjects o,
  2043.             master.dbo.spt_datatype_info d,
  2044.             systypes t
  2045.         WHERE
  2046.             o.id = @procedure_id
  2047.             AND c.id = o.id
  2048.             AND c.type = d.ss_dtype
  2049.             AND c.usertype *= t.usertype
  2050.             AND c.name like @column_name
  2051.             AND c.number = @group_num
  2052.     end
  2053.     else
  2054.     begin
  2055.         /* this block is for the case where there IS pattern
  2056.             matching done on the procedure name */
  2057.         if @procedure_owner is null
  2058.             select @procedure_owner = '%'
  2059.         SELECT
  2060.             procedure_qualifier = DB_NAME(),
  2061.             procedure_owner = USER_NAME(o.uid),
  2062.             procedure_name = o.name +';'+ ltrim(str(c.number,5)),
  2063.             column_name = c.name,
  2064.             column_type = convert(smallint, 0),
  2065.             data_type = d.data_type+convert(smallint,
  2066.                         isnull(d.aux,
  2067.                         ascii(substring("666AAA@@@CB??GG",
  2068.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))
  2069.                         -60)),
  2070.             type_name = rtrim(substring(d.type_name,
  2071.                         1+isnull(d.aux,
  2072.                         ascii(substring("III<<<MMMI<<A<A",
  2073.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2074.                         1))-60), 13)),
  2075.             precision = isnull(d.data_precision, convert(int,c.length))
  2076.                         +isnull(d.aux, convert(int,
  2077.                         ascii(substring("???AAAFFFCKFOLS",
  2078.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  2079.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  2080.                         isnull(d.aux,
  2081.                         ascii(substring("AAA<BB<DDDHJSPP",
  2082.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2083.                         1))-64)),
  2084.             scale = d.numeric_scale +convert(smallint,
  2085.                         isnull(d.aux,
  2086.                         ascii(substring("<<<<<<<<<<<<<<?",
  2087.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2088.                         1))-60)),
  2089.             radix = d.numeric_radix,
  2090.             nullable = convert(smallint,t.allownulls),
  2091.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  2092.             ss_data_type = c.type,
  2093.             c.colid
  2094.         FROM
  2095.             syscolumns c,
  2096.             sysobjects o,
  2097.             master.dbo.spt_datatype_info d,
  2098.             systypes t
  2099.         WHERE
  2100.             o.name like @procedure_name
  2101.             AND user_name(o.uid) like @procedure_owner
  2102.             AND c.id = o.id
  2103.             AND c.usertype *= t.usertype
  2104.             AND c.type = d.ss_dtype
  2105.             AND c.name like @column_name
  2106.             AND o.type = 'P'    /* Just Procedures */
  2107.         ORDER BY procedure_qualifier, procedure_owner, procedure_name, colid
  2108.     end
  2109. go
  2110.  
  2111.  
  2112. /* if this is a 4.2 or later SQL Server, then we want to delete the stored
  2113. procedure we just created, and create a new one which uses UNION
  2114. (this fixes some bugs involving UDT names and char NULL datatype
  2115. names)
  2116. */
  2117.  
  2118. if  (charindex('1.1', @@version) = 0
  2119.     and charindex('4.0', @@version) = 0)
  2120. begin
  2121.     drop proc sp_sproc_columns
  2122.     dump tran master with truncate_only
  2123. end
  2124. else
  2125. begin
  2126.     print ""
  2127.     print ""
  2128.     print "Installing catalog procedures on a 1.x or 4.0 server:"
  2129.     print "Ignore the following error messages"
  2130.     print ""
  2131.     print ""
  2132. end
  2133. go
  2134.  
  2135.  
  2136.  
  2137. /* this is the version for servers which support UNION */
  2138. CREATE PROCEDURE sp_sproc_columns (
  2139.                  @procedure_name        varchar(36) = '%',
  2140.                  @procedure_owner        varchar(32) = null,
  2141.                  @procedure_qualifier    varchar(32) = null,
  2142.                  @column_name            varchar(32) = null )
  2143. AS
  2144.     DECLARE @group_num int
  2145.     DECLARE @semi_position int
  2146.     DECLARE @full_procedure_name    char(70)
  2147.     DECLARE @procedure_id int
  2148.  
  2149.     if @column_name is null /*    If column name not supplied, match all */
  2150.         select @column_name = '%'
  2151.     if @procedure_qualifier is not null
  2152.     begin
  2153.         if db_name() != @procedure_qualifier
  2154.         begin
  2155.             if @procedure_qualifier = ''
  2156.             begin
  2157.                 /* in this case, we need to return an empty result set */
  2158.                 /* because the user has requested a database with an empty name */
  2159.                 select @procedure_name = ''
  2160.                 select @procedure_owner = ''
  2161.             end
  2162.             else
  2163.             begin    /* If qualifier doesn't match current database */
  2164.                 print "Procedure qualifier must be name of current database"
  2165.                 return
  2166.             end
  2167.         end
  2168.     end
  2169.  
  2170.  
  2171.     /* first we need to extract the procedure group number, if one exists */
  2172.     select @semi_position = charindex(';',@procedure_name)
  2173.     if (@semi_position > 0)
  2174.     begin    /* If group number separator (;) found */
  2175.         select @group_num = convert(int,substring(@procedure_name, @semi_position + 1, 2))
  2176.         select @procedure_name = substring(@procedure_name, 1, @semi_position -1)
  2177.     end
  2178.     else
  2179.     begin    /* No group separator, so default to group number of 1 */
  2180.         select @group_num = 1
  2181.     end
  2182.  
  2183.     if @procedure_owner is null
  2184.     begin    /* If unqualified procedure name */
  2185.         SELECT @full_procedure_name = @procedure_name
  2186.     end
  2187.     else
  2188.     begin    /* Qualified procedure name */
  2189.         SELECT @full_procedure_name = @procedure_owner + '.' + @procedure_name
  2190.     end
  2191.  
  2192.     /*    Get Object ID */
  2193.     SELECT @procedure_id = object_id(@full_procedure_name)
  2194.     if ((charindex('%',@full_procedure_name) = 0) and
  2195.         (charindex('_',@full_procedure_name) = 0)  and
  2196.         @procedure_id != 0)
  2197.     begin
  2198.         /* this block is for the case where there is no pattern
  2199.             matching required for the table name */
  2200.         SELECT    /* INTn, FLOATn, DATETIMEn and MONEYn types */
  2201.             procedure_qualifier = DB_NAME(),
  2202.             procedure_owner = USER_NAME(o.uid),
  2203.             procedure_name = o.name +';'+ ltrim(str(c.number,5)),
  2204.             column_name = c.name,
  2205.             column_type = convert(smallint, 0),
  2206.             data_type = d.data_type+convert(smallint,
  2207.                         isnull(d.aux,
  2208.                         ascii(substring("666AAA@@@CB??GG",
  2209.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))
  2210.                         -60)),
  2211.             type_name = rtrim(substring(d.type_name,
  2212.                         1+isnull(d.aux,
  2213.                         ascii(substring("III<<<MMMI<<A<A",
  2214.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2215.                         1))-60), 13)),
  2216.             precision = isnull(d.data_precision, convert(int,c.length))
  2217.                         +isnull(d.aux, convert(int,
  2218.                         ascii(substring("???AAAFFFCKFOLS",
  2219.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  2220.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  2221.                         isnull(d.aux,
  2222.                         ascii(substring("AAA<BB<DDDHJSPP",
  2223.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2224.                         1))-64)),
  2225.             scale = d.numeric_scale +convert(smallint,
  2226.                         isnull(d.aux,
  2227.                         ascii(substring("<<<<<<<<<<<<<<?",
  2228.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2229.                         1))-60)),
  2230.             radix = d.numeric_radix,
  2231.             nullable = convert(smallint,t.allownulls),
  2232.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  2233.             ss_data_type = c.type,
  2234.             colid = c.colid
  2235.         FROM
  2236.             syscolumns c,
  2237.             sysobjects o,
  2238.             master.dbo.spt_datatype_info d,
  2239.             systypes t,
  2240.             sysprocedures p
  2241.         WHERE
  2242.             o.id = @procedure_id
  2243.             AND c.id = o.id
  2244.             AND c.usertype = t.usertype
  2245.             AND c.type = d.ss_dtype
  2246.             AND c.name like @column_name
  2247.             AND d.ss_dtype IN (111, 109, 38, 110)    /* Just *N types */
  2248.             AND c.number = @group_num
  2249.         UNION
  2250.         SELECT           /* All other types including user data types */
  2251.             procedure_qualifier = DB_NAME(),
  2252.             procedure_owner = USER_NAME(o.uid),
  2253.             procedure_name = o.name +';'+ ltrim(str(c.number,5)),
  2254.             column_name = c.name,
  2255.             column_type = convert(smallint, 0),
  2256.             data_type = convert(smallint,
  2257.                 /*    Map systypes.type to ODBC type */
  2258.                 /*        SS-Type         "                 1          "*/
  2259.                 /*                        "33 3 3 4 44 5 5 2 5 55666"*/
  2260.                 /*                        "45 7 9    5 78 0 2 2 6 89012"*/
  2261.                         ascii(substring("8;<9<H<<<<<:<=6<5<A<?<@<GC?GD",
  2262.                         t.type%34+1,1))-60),
  2263.             type_name = t.name,
  2264.             precision = isnull(d.data_precision, convert(int,c.length))
  2265.                         +isnull(d.aux, convert(int,
  2266.                         ascii(substring("???AAAFFFCKFOLS",
  2267.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  2268.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  2269.                         isnull(d.aux,
  2270.                         ascii(substring("AAA<BB<DDDHJSPP",
  2271.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2272.                         1))-64)),
  2273.             scale = d.numeric_scale +convert(smallint,
  2274.                         isnull(d.aux,
  2275.                         ascii(substring("<<<<<<<<<<<<<<?",
  2276.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2277.                         1))-60)),
  2278.             radix = d.numeric_radix,
  2279.             nullable = convert(smallint,t.allownulls),
  2280.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  2281.             ss_data_type = c.type,
  2282.             colid = c.colid
  2283.         FROM
  2284.             syscolumns c,
  2285.             sysobjects o,
  2286.             master.dbo.spt_datatype_info d,
  2287.             systypes t
  2288.         WHERE
  2289.             o.id = @procedure_id
  2290.             AND c.id = o.id
  2291.             AND c.type = d.ss_dtype
  2292.             AND c.usertype *= t.usertype
  2293.             AND c.name like @column_name
  2294.             AND d.ss_dtype NOT IN (111, 109, 38, 110)    /* No *N types */
  2295.             AND c.number = @group_num
  2296.          ORDER BY colid
  2297.     end
  2298.     else
  2299.     begin
  2300.         /* this block is for the case where there IS pattern
  2301.             matching done on the table name */
  2302.         if @procedure_owner is null
  2303.             select @procedure_owner = '%'
  2304.         SELECT    /* INTn, FLOATn, DATETIMEn and MONEYn types */
  2305.             procedure_qualifier = DB_NAME(),
  2306.             procedure_owner = USER_NAME(o.uid),
  2307.             procedure_name = o.name +';'+ ltrim(str(c.number,5)),
  2308.             column_name = c.name,
  2309.             column_type = convert(smallint, 0),
  2310.             data_type = d.data_type+convert(smallint,
  2311.                         isnull(d.aux,
  2312.                         ascii(substring("666AAA@@@CB??GG",
  2313.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))
  2314.                         -60)),
  2315.             type_name = rtrim(substring(d.type_name,
  2316.                         1+isnull(d.aux,
  2317.                         ascii(substring("III<<<MMMI<<A<A",
  2318.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2319.                         1))-60), 13)),
  2320.             precision = isnull(d.data_precision, convert(int,c.length))
  2321.                         +isnull(d.aux, convert(int,
  2322.                         ascii(substring("???AAAFFFCKFOLS",
  2323.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  2324.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  2325.                         isnull(d.aux,
  2326.                         ascii(substring("AAA<BB<DDDHJSPP",
  2327.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2328.                         1))-64)),
  2329.             scale = d.numeric_scale +convert(smallint,
  2330.                         isnull(d.aux,
  2331.                         ascii(substring("<<<<<<<<<<<<<<?",
  2332.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2333.                         1))-60)),
  2334.             radix = d.numeric_radix,
  2335.             nullable = convert(smallint,t.allownulls),
  2336.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  2337.             ss_data_type = c.type,
  2338.             colid = c.colid
  2339.         FROM
  2340.             syscolumns c,
  2341.             sysobjects o,
  2342.             master.dbo.spt_datatype_info d,
  2343.             systypes t
  2344.         WHERE
  2345.             o.name like @procedure_name
  2346.             AND user_name(o.uid) like @procedure_owner
  2347.             AND o.id = c.id
  2348.             AND c.usertype = t.usertype
  2349.             AND c.type = d.ss_dtype
  2350.             AND c.name like @column_name
  2351.             AND o.type = 'P'                        /* Just Procedures */
  2352.             AND d.ss_dtype IN (111, 109, 38, 110)    /* Just *N types */
  2353.         UNION
  2354.         SELECT           /* All other types including user data types */
  2355.             procedure_qualifier = DB_NAME(),
  2356.             procedure_owner = USER_NAME(o.uid),
  2357.             procedure_name = o.name +';'+ ltrim(str(c.number,5)),
  2358.             column_name = c.name,
  2359.             column_type = convert(smallint, 0),
  2360.             data_type = convert(smallint,
  2361.                 /*    Map systypes.type to ODBC type */
  2362.                 /*        SS-Type         "                 1          "*/
  2363.                 /*                        "33 3 3 4 44 5 5 2 5 55666"*/
  2364.                 /*                        "45 7 9    5 78 0 2 2 6 89012"*/
  2365.                         ascii(substring("8;<9<H<<<<<:<=6<5<A<?<@<GC?GD",
  2366.                         t.type%34+1,1))-60),
  2367.             type_name = t.name,
  2368.             precision = isnull(d.data_precision, convert(int,c.length))
  2369.                         +isnull(d.aux, convert(int,
  2370.                         ascii(substring("???AAAFFFCKFOLS",
  2371.                         2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  2372.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  2373.                         isnull(d.aux,
  2374.                         ascii(substring("AAA<BB<DDDHJSPP",
  2375.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2376.                         1))-64)),
  2377.             scale = d.numeric_scale +convert(smallint,
  2378.                         isnull(d.aux,
  2379.                         ascii(substring("<<<<<<<<<<<<<<?",
  2380.                         2*(d.ss_dtype%35+1)+2-8/c.length,
  2381.                         1))-60)),
  2382.             radix = d.numeric_radix,
  2383.             nullable = convert(smallint,t.allownulls),
  2384.             remarks = convert(varchar(254),null),    /* Remarks are NULL */
  2385.             ss_data_type = c.type,
  2386.             colid = c.colid
  2387.         FROM
  2388.             syscolumns c,
  2389.             sysobjects o,
  2390.             master.dbo.spt_datatype_info d,
  2391.             systypes t
  2392.         WHERE
  2393.             o.name like @procedure_name
  2394.             AND user_name(o.uid) like @procedure_owner
  2395.             AND o.id = c.id
  2396.             AND c.type = d.ss_dtype
  2397.             AND c.usertype *= t.usertype
  2398.             AND o.type = 'P'                            /* Just Procedures */
  2399.             AND c.name like @column_name
  2400.             AND d.ss_dtype NOT IN (111, 109, 38, 110)    /* No *N types */
  2401.         ORDER BY procedure_owner, procedure_name, colid
  2402.     end
  2403. go
  2404.  
  2405. grant execute on sp_sproc_columns to public
  2406. go
  2407.  
  2408. print "creating sp_table_privileges"
  2409. go
  2410.  
  2411. CREATE PROCEDURE sp_table_privileges (
  2412.             @table_name         varchar(32),
  2413.             @table_owner        varchar(32) = null,
  2414.             @table_qualifier    varchar(32) = null)
  2415. as
  2416.  
  2417.     declare @table_id    int,
  2418.             @owner_id     int,
  2419.             @full_table_name char(70)
  2420.  
  2421.     if @table_qualifier is not null
  2422.     begin
  2423.         if db_name() != @table_qualifier
  2424.         begin    /* If qualifier doesn't match current database */
  2425.             print "Table qualifier must be name of current database"
  2426.             return
  2427.         end
  2428.     end
  2429.     if @table_owner is null
  2430.     begin    /* If unqualified table name */
  2431.         SELECT @full_table_name = @table_name
  2432.     end
  2433.     else
  2434.     begin    /* Qualified table name */
  2435.         SELECT @full_table_name = @table_owner + '.' + @table_name
  2436.     end
  2437.     /*    Get Object ID */
  2438.     SELECT @table_id = object_id(@full_table_name)
  2439.  
  2440.     if @@trancount != 0
  2441.     begin    /* If inside a transaction */
  2442.         raiserror 20001 "catalog procedure sp_table_privileges can not be run in a transaction"
  2443.         return
  2444.     end
  2445.     create table #table_priv1(
  2446.         table_qualifier        varchar(32),
  2447.         table_owner         varchar(32),
  2448.         table_name            varchar(32),
  2449.         grantor             varchar(32),
  2450.         grantee             varchar(32),
  2451.         select_privilege    int,
  2452.         insert_privilege    int,
  2453.         update_privilege    int,
  2454.         delete_privilege    int,
  2455.         is_grantable        varchar(3),
  2456.         uid                 int,
  2457.         gid                 int)
  2458.  
  2459.     insert into #table_priv1
  2460.         select distinct
  2461.             db_name(),
  2462.             user_name(o.uid),
  2463.             o.name,
  2464.             user_name(o.uid),
  2465.             u.name,
  2466.             0,
  2467.             0,
  2468.             0,
  2469.             0,
  2470.             'no',
  2471.             u.uid,
  2472.             u.gid
  2473.         from sysusers u, sysobjects o
  2474.         where o.id = @table_id and u.uid != u.gid
  2475.            and sysstat & 7 in (1,2,3)    /* only valid for system tables,
  2476.                                            ** user tables, and views. */
  2477.  
  2478.     /*
  2479.     ** now add row for table owner
  2480.     */
  2481.     if exists (
  2482.         select *
  2483.             from #table_priv1
  2484.             where grantor = grantee)
  2485.     begin
  2486.         update #table_priv1
  2487.         set
  2488.             select_privilege = 1,
  2489.             update_privilege = 1,
  2490.             insert_privilege = 1,
  2491.             delete_privilege = 1,
  2492.             is_grantable = 'yes'
  2493.         where grantor = grantee
  2494.     end
  2495.     else
  2496.     begin
  2497.         insert into #table_priv1
  2498.             select    db_name(),
  2499.                 user_name(o.uid),
  2500.                 o.name,
  2501.                 user_name(o.uid),
  2502.                 user_name(o.uid),
  2503.                 1,
  2504.                 1,
  2505.                 1,
  2506.                 1,
  2507.                 'yes',
  2508.                 o.uid,
  2509.                 u.gid
  2510.             from sysobjects o, sysusers u
  2511.             where o.id = @table_id and u.uid = o.uid
  2512.             and sysstat & 7 in (1,2,3)    /* only valid for system tables,
  2513.                                            ** user tables, and views. */
  2514.  
  2515.     end
  2516.  
  2517.     update #table_priv1
  2518.     set select_privilege = 1
  2519.     where
  2520.         exists (
  2521.             select * from sysprotects
  2522.             where
  2523.                 id = @table_id
  2524.                 and (#table_priv1.uid = uid
  2525.                     or #table_priv1.gid = uid
  2526.                     or uid = 0)
  2527.                 and protecttype = 205
  2528.                 and action = 193)
  2529.         and not exists (
  2530.             select * from sysprotects
  2531.             where
  2532.                 id = @table_id
  2533.                 and (#table_priv1.uid = uid
  2534.                     or #table_priv1.gid = uid
  2535.                     or uid = 0)
  2536.                 and protecttype = 206
  2537.                 and action = 193)
  2538.  
  2539.     update #table_priv1
  2540.     set insert_privilege = 1
  2541.     where
  2542.         exists (
  2543.             select * from sysprotects
  2544.             where
  2545.                 id = @table_id
  2546.                 and (#table_priv1.uid = uid
  2547.                     or #table_priv1.gid = uid
  2548.                     or uid = 0)
  2549.                 and protecttype = 205
  2550.                 and action = 195)
  2551.         and not exists (
  2552.             select * from sysprotects
  2553.             where
  2554.                 id = @table_id
  2555.                 and (#table_priv1.uid = uid
  2556.                     or #table_priv1.gid = uid
  2557.                     or uid = 0)
  2558.                 and protecttype = 206
  2559.                 and action = 195)
  2560.  
  2561.     update #table_priv1
  2562.     set delete_privilege = 1
  2563.     where
  2564.         exists (
  2565.             select * from sysprotects
  2566.             where
  2567.                 id = @table_id
  2568.                 and (#table_priv1.uid = uid
  2569.                     or #table_priv1.gid = uid
  2570.                     or uid = 0)
  2571.                 and protecttype = 205
  2572.                 and action = 196)
  2573.         and not exists (select * from sysprotects
  2574.             where
  2575.                 id = @table_id
  2576.                 and (#table_priv1.uid = uid
  2577.                     or #table_priv1.gid = uid
  2578.                     or uid = 0)
  2579.                 and protecttype = 206
  2580.                 and action = 196)
  2581.  
  2582.     update #table_priv1
  2583.     set update_privilege = 1
  2584.     where
  2585.         exists (
  2586.             select * from sysprotects
  2587.             where
  2588.                 id = @table_id
  2589.                 and (#table_priv1.uid = uid
  2590.                     or #table_priv1.gid = uid
  2591.                     or uid = 0)
  2592.                 and protecttype = 205
  2593.                 and action = 197)
  2594.         and not exists (
  2595.             select * from sysprotects
  2596.             where
  2597.                 id = @table_id
  2598.                 and (#table_priv1.uid = uid
  2599.                     or #table_priv1.gid = uid
  2600.                     or uid = 0)
  2601.                 and protecttype = 206
  2602.                 and action = 197)
  2603.  
  2604.     create table #table_priv2(
  2605.         table_qualifier varchar(32),
  2606.         table_owner     varchar(32),
  2607.         table_name        varchar(32),
  2608.         grantor         varchar(32),
  2609.         grantee         varchar(32),
  2610.         privilege        varchar(32),
  2611.         is_grantable    varchar(3))
  2612.  
  2613.     insert into #table_priv2
  2614.         select
  2615.             table_qualifier,
  2616.             table_owner,
  2617.             table_name,
  2618.             grantor,
  2619.             grantee,
  2620.             'SELECT',
  2621.             is_grantable
  2622.         from #table_priv1
  2623.         where select_privilege = 1
  2624.  
  2625.  
  2626.     insert into #table_priv2
  2627.         select
  2628.             table_qualifier,
  2629.             table_owner,
  2630.             table_name,
  2631.             grantor,
  2632.             grantee,
  2633.             'INSERT',
  2634.             is_grantable
  2635.         from #table_priv1
  2636.         where insert_privilege = 1
  2637.  
  2638.  
  2639.     insert into #table_priv2
  2640.         select
  2641.             table_qualifier,
  2642.             table_owner,
  2643.             table_name,
  2644.             grantor,
  2645.             grantee,
  2646.             'DELETE',
  2647.             is_grantable
  2648.         from #table_priv1
  2649.         where delete_privilege = 1
  2650.  
  2651.  
  2652.     insert into #table_priv2
  2653.         select
  2654.             table_qualifier,
  2655.             table_owner,
  2656.             table_name,
  2657.             grantor,
  2658.             grantee,
  2659.             'UPDATE',
  2660.             is_grantable
  2661.         from #table_priv1
  2662.         where update_privilege = 1
  2663.  
  2664.  
  2665.     select * from #table_priv2
  2666.     order by privilege
  2667. go
  2668.  
  2669. grant execute on sp_table_privileges to public
  2670. go
  2671.  
  2672. dump tran master with truncate_only
  2673. go
  2674.  
  2675. print "creating sp_column_privileges"
  2676. go
  2677.  
  2678.  
  2679. CREATE PROCEDURE sp_column_privileges (
  2680.             @table_name         varchar(32),
  2681.             @table_owner        varchar(32) = null,
  2682.             @table_qualifier    varchar(32) = null,
  2683.             @column_name        varchar(32) = null)
  2684. as
  2685.  
  2686.     declare @table_id    int,
  2687.     @owner_id    int
  2688.     DECLARE @full_table_name    char(70)
  2689.     declare @low int                    /* range of userids to check */
  2690.     declare @high int
  2691.     declare @objid int                  /* id of @name if object */
  2692.     declare @owner_name varchar(32)
  2693.  
  2694.     select @low = 0, @high = 32767
  2695.  
  2696.     if @column_name is null /*    If column name not supplied, match all */
  2697.         select @column_name = '%'
  2698.  
  2699.     if @table_qualifier is not null
  2700.     begin
  2701.         if db_name() != @table_qualifier
  2702.         begin    /* If qualifier doesn't match current database */
  2703.             print "Table qualifier must be name of current database"
  2704.             return
  2705.         end
  2706.     end
  2707.     if @table_owner is null
  2708.     begin    /* If unqualified table name */
  2709.         SELECT @full_table_name = @table_name
  2710.     end
  2711.     else
  2712.     begin    /* Qualified table name */
  2713.         SELECT @full_table_name = @table_owner + '.' + @table_name
  2714.     end
  2715.     /*    Get Object ID */
  2716.     select @table_id = object_id(@full_table_name)
  2717.  
  2718.     if @@trancount != 0
  2719.     begin    /* If inside a transaction */
  2720.         raiserror 20001 "catalog procedure sp_column_privileges can not be run in a transaction"
  2721.         return
  2722.     end
  2723.  
  2724.     /*
  2725.     ** We need to create a table which will contain a row for every row to
  2726.     ** be returned to the client.
  2727.     */
  2728.  
  2729.     create table #column_priv1(
  2730.         table_qualifier            varchar(32),
  2731.         table_owner             varchar(32),
  2732.         table_name                varchar(32),
  2733.         column_name             varchar(32),
  2734.         grantor                 varchar(32),
  2735.         grantee                 varchar(32),
  2736.         select_privilege        int,
  2737.         select_grantable        int,
  2738.         insert_privilege        int,
  2739.         insert_grantable        int,
  2740.         update_privilege        int,
  2741.         update_grantable        int,
  2742.         delete_privilege        int,
  2743.         delete_grantable        int,
  2744.         references_privilege    int null,
  2745.         references_grantable    int null,
  2746.         uid                     int,
  2747.         gid                     int,
  2748.         is_grantable            varchar(3))
  2749.  
  2750. /*
  2751. ** insert a row for the table owner (who has all permissions)
  2752. */
  2753.     select @owner_name = (
  2754.         select user_name(uid)
  2755.         from sysobjects
  2756.         where id = @table_id)
  2757.  
  2758.     insert into #column_priv1
  2759.         select
  2760.             db_name(),
  2761.             @owner_name,
  2762.             @table_name,
  2763.             name,
  2764.             @owner_name,
  2765.             @owner_name,
  2766.             1,
  2767.             1,
  2768.             1,
  2769.             1,
  2770.             1,
  2771.             1,
  2772.             1,
  2773.             1,
  2774.             null,
  2775.             null,
  2776.             user_id(@owner_name),
  2777.             0,
  2778.             'yes'
  2779.         from syscolumns
  2780.         where id = @table_id
  2781.  
  2782. /*
  2783. ** now stick a row in the table for every user in the database
  2784. ** we will need to weed out those who have no permissions later
  2785. ** (and yes this is a cartesion product: the uid field in sysprotects
  2786. ** can also have a group id, in which case we need to extend those
  2787. ** privileges to all group members).
  2788. */
  2789.  
  2790.     insert into #column_priv1
  2791.         select distinct
  2792.             db_name(),
  2793.             user_name(o.uid),
  2794.             @table_name,
  2795.             c.name,
  2796.             user_name(o.uid),
  2797.             u.name,
  2798.             0,
  2799.             0,
  2800.             0,
  2801.             0,
  2802.             0,
  2803.             0,
  2804.             0,
  2805.             0,
  2806.             null,
  2807.             null,
  2808.             u.uid,
  2809.             u.gid,
  2810.             'no'
  2811.         from sysusers u, syscolumns c, sysobjects o
  2812.         where o.id = @table_id
  2813.             and c.id = o.id
  2814.             and u.gid != u.uid
  2815.             and u.name != @owner_name
  2816.  
  2817.     /*
  2818.     ** we need to create another temporary table to contain all the various
  2819.     ** protection information for the table in question
  2820.     */
  2821.     create table #protects (
  2822.                 uid         smallint,
  2823.                 action        tinyint,
  2824.                 protecttype tinyint,
  2825.                 name        varchar(32))
  2826.  
  2827.     insert into #protects
  2828.         select
  2829.             p.uid,
  2830.             p.action,
  2831.             p.protecttype,
  2832.             isnull(col_name(id, c.number), "All")
  2833.             from
  2834.                 sysprotects p,
  2835.                 master.dbo.spt_values c,
  2836.                 master.dbo.spt_values a,
  2837.                 master.dbo.spt_values b
  2838.             where
  2839.                 convert(tinyint, substring(isnull(p.columns, 0x1), c.low, 1))
  2840.                     & c.high != 0
  2841.                     and c.number <= (
  2842.                         select count(*)
  2843.                         from syscolumns
  2844.                         where id = @table_id)
  2845.                 and a.type = "T"
  2846.                 and a.number = p.action
  2847.                 and b.type = "T"
  2848.                 and b.number = p.protecttype
  2849.                 and p.id = @table_id
  2850.                 and p.uid between @low and @high
  2851.  
  2852.  
  2853.     update #column_priv1
  2854.     set select_privilege = 1
  2855.     where
  2856.         exists (
  2857.             select * from #protects
  2858.             where
  2859.                 protecttype = 205
  2860.                 and action = 193
  2861.                 and (name = #column_priv1.column_name
  2862.                     or name = 'All')
  2863.                 and (uid = 0
  2864.                     or uid = #column_priv1.gid
  2865.                     or uid = #column_priv1.uid))
  2866.         and not exists (
  2867.             select * from #protects
  2868.             where
  2869.                 protecttype = 206
  2870.                 and action = 193
  2871.                 and (name = #column_priv1.column_name
  2872.                     or name = 'All')
  2873.                 and ( uid = 0
  2874.                     or uid = #column_priv1.gid
  2875.                     or uid = #column_priv1.uid))
  2876.  
  2877.     update #column_priv1
  2878.     set insert_privilege = 1
  2879.     where
  2880.         exists (
  2881.             select * from #protects
  2882.             where
  2883.                 protecttype = 205
  2884.                 and action = 195
  2885.                 and (name = #column_priv1.column_name
  2886.                     or name = 'All')
  2887.                 and (uid = 0
  2888.                     or uid = #column_priv1.gid
  2889.                     or uid = #column_priv1.uid))
  2890.         and not exists (
  2891.             select * from #protects
  2892.             where
  2893.                 protecttype = 206
  2894.                 and action = 195
  2895.                 and (name = #column_priv1.column_name
  2896.                     or name = 'All')
  2897.                 and (uid = 0
  2898.                     or uid = #column_priv1.gid
  2899.                     or uid = #column_priv1.uid))
  2900.  
  2901.     update #column_priv1
  2902.     set insert_privilege = 1
  2903.     where
  2904.         exists (
  2905.             select * from #protects
  2906.             where
  2907.                 protecttype = 205
  2908.                 and action = 196
  2909.                 and (name = #column_priv1.column_name
  2910.                     or name = 'All')
  2911.                 and (uid = 0
  2912.                     or uid = #column_priv1.gid
  2913.                     or uid = #column_priv1.uid))
  2914.         and not exists (
  2915.             select * from #protects
  2916.             where
  2917.                 protecttype = 206
  2918.                 and action = 196
  2919.                 and (name = #column_priv1.column_name
  2920.                     or name = 'All')
  2921.                 and (uid = 0
  2922.                     or uid = #column_priv1.gid
  2923.                     or uid = #column_priv1.uid))
  2924.  
  2925.  
  2926.     update #column_priv1
  2927.     set update_privilege = 1
  2928.         where
  2929.             exists (
  2930.                 select * from #protects
  2931.                 where protecttype = 205
  2932.                 and action = 197
  2933.                 and (name = #column_priv1.column_name
  2934.                     or name = 'All')
  2935.                 and (uid = 0
  2936.                     or uid = #column_priv1.gid
  2937.                     or uid = #column_priv1.uid))
  2938.             and not exists (
  2939.                 select * from #protects
  2940.                     where protecttype = 206
  2941.                     and action = 197
  2942.                     and (name = #column_priv1.column_name
  2943.                         or name = 'All')
  2944.                     and (uid = 0
  2945.                         or uid = #column_priv1.gid
  2946.                         or uid = #column_priv1.uid))
  2947.  
  2948.     create table #column_priv2(
  2949.         table_qualifier varchar(32),
  2950.         table_owner     varchar(32),
  2951.         table_name        varchar(32),
  2952.         column_name     varchar(32),
  2953.         grantor         varchar(32),
  2954.         grantee         varchar(32),
  2955.         privilege        varchar(32),
  2956.         is_grantable    varchar(3))
  2957.  
  2958.     insert into #column_priv2
  2959.         select
  2960.             table_qualifier,
  2961.             table_owner,
  2962.             table_name,
  2963.             column_name,
  2964.             grantor,
  2965.             grantee,
  2966.             'SELECT',
  2967.             is_grantable
  2968.         from #column_priv1
  2969.         where select_privilege = 1
  2970.  
  2971.     insert into #column_priv2
  2972.         select
  2973.             table_qualifier,
  2974.             table_owner,
  2975.             table_name,
  2976.             column_name,
  2977.             grantor,
  2978.             grantee,
  2979.             'INSERT',
  2980.             is_grantable
  2981.         from #column_priv1
  2982.         where insert_privilege = 1
  2983.  
  2984.     insert into #column_priv2
  2985.         select
  2986.             table_qualifier,
  2987.             table_owner,
  2988.             table_name,
  2989.             column_name,
  2990.             grantor,
  2991.             grantee,
  2992.             'UPDATE',
  2993.             is_grantable
  2994.         from #column_priv1
  2995.         where update_privilege = 1
  2996.  
  2997.     select * from #column_priv2
  2998.     where column_name like @column_name
  2999.     order by column_name, privilege
  3000. go
  3001.  
  3002. grant execute on sp_column_privileges to public
  3003. go
  3004.  
  3005. dump tran master with truncate_only
  3006. go
  3007.  
  3008.  
  3009. print "creating sp_server_info"
  3010. go
  3011.  
  3012. create proc sp_server_info (
  3013.             @attribute_id  int = null)
  3014. as
  3015.     if @attribute_id is not null
  3016.         select *
  3017.         from master.dbo.spt_server_info
  3018.         where attribute_id = @attribute_id
  3019.     else
  3020.         select *
  3021.         from master.dbo.spt_server_info
  3022.         order by attribute_id
  3023. go
  3024.  
  3025. grant execute on sp_server_info to public
  3026. go
  3027.  
  3028. print "creating sp_datatype_info"
  3029. go
  3030.  
  3031. /* the messiness of 'data_type' was to get around the problem of
  3032. returning the correct lengths for user defined types.  the join
  3033. on the type name ensures all user defined types are returned, but
  3034. this puts a null in the data_type column.  by forcing an embedded
  3035. select and correlating it with the current row in systypes, we get
  3036. the correct data_type mapping even for user defined types  (kwg) */
  3037.  
  3038. /* this is the version for servers without support for UNION */
  3039. create proc sp_datatype_info (
  3040.             @data_type int = 0)
  3041. as
  3042.     if @data_type = 0
  3043.         select
  3044.             type_name = t.name,
  3045.             d.data_type,
  3046.             precision = isnull(d.data_precision, convert(int,t.length)),
  3047.             d.literal_prefix,
  3048.             d.literal_suffix,
  3049.             e.create_params,
  3050.             d.nullable,
  3051.             d.case_sensitive,
  3052.             d.searchable,
  3053.             d.unsigned_attribute,
  3054.             d.money,
  3055.             d.auto_increment,
  3056.             d.local_type_name
  3057.         from master.dbo.spt_datatype_info d, master.dbo.spt_datatype_info_ext e, systypes t
  3058.         where
  3059.             d.ss_dtype = t.type
  3060.             and t.usertype *= e.user_type
  3061.             and t.type not in (111,109,38,110)    /* get rid of nullable types */
  3062.         order by d.data_type, type_name
  3063.     else
  3064.         select
  3065.             type_name = t.name,
  3066.             d.data_type,
  3067.             precision = isnull(d.data_precision, convert(int,t.length)),
  3068.             d.literal_prefix,
  3069.             d.literal_suffix,
  3070.             e.create_params,
  3071.             d.nullable,
  3072.             d.case_sensitive,
  3073.             d.searchable,
  3074.             d.unsigned_attribute,
  3075.             d.money,
  3076.             d.auto_increment,
  3077.             d.local_type_name
  3078.         from
  3079.             master.dbo.spt_datatype_info d,
  3080.             master.dbo.spt_datatype_info_ext e,
  3081.             systypes t
  3082.         where
  3083.             data_type = @data_type
  3084.             and    d.ss_dtype = t.type
  3085.             and t.usertype *= e.user_type
  3086.             and t.type not in (111,109,38,110)    /* get rid of nullable types */
  3087.         order by type_name
  3088. go
  3089.  
  3090. /* if this is a 4.2 or later SQL Server, then we want to delete the stored
  3091. procedure we just created, and create a new one which uses UNION
  3092. (this fixes some bugs involving UDT names)
  3093. */
  3094.  
  3095. if  (charindex('1.1', @@version) = 0
  3096.     and charindex('4.0', @@version) = 0)
  3097. begin
  3098.     drop proc sp_datatype_info
  3099.     dump tran master with truncate_only
  3100. end
  3101. else
  3102. begin
  3103.     print ""
  3104.     print ""
  3105.     print "Installing catalog procedures on a 1.x or 4.0 server:"
  3106.     print "Ignore the following error messages"
  3107.     print ""
  3108.     print ""
  3109. end
  3110. go
  3111.  
  3112. /* this is the version for servers which support UNION */
  3113. create proc sp_datatype_info (
  3114.             @data_type int = 0)
  3115. as
  3116.     if @data_type = 0
  3117.         select    /* Real SQL Server data types */
  3118.             type_name = t.name,
  3119.             d.data_type,
  3120.             precision = isnull(d.data_precision, convert(int,t.length)),
  3121.             d.literal_prefix,
  3122.             d.literal_suffix,
  3123.             e.create_params,
  3124.             d.nullable,
  3125.             d.case_sensitive,
  3126.             d.searchable,
  3127.             d.unsigned_attribute,
  3128.             d.money,
  3129.             d.auto_increment,
  3130.             d.local_type_name
  3131.         from master.dbo.spt_datatype_info d, master.dbo.spt_datatype_info_ext e, systypes t
  3132.         where
  3133.             d.ss_dtype = t.type
  3134.             and t.usertype *= e.user_type
  3135.             and t.usertype not in (80,18)        /* No SYSNAME or TIMESTAMP */
  3136.             and t.usertype < 100                /* No user defined types */
  3137.             and t.type not in (111,109,38,110)    /* get rid of nullable types */
  3138.             and t.name != 'nchar'                /* get rid of 4.9.1 type */
  3139.             and t.name != 'nvarchar'            /* get rid of 4.9.1 type */
  3140.         UNION
  3141.         select    /* SQL Server SYSNAME, TIMESTAMP and user data types */
  3142.             type_name = t.name,
  3143.             d.data_type,
  3144.             precision = isnull(d.data_precision, convert(int,t.length)),
  3145.             d.literal_prefix,
  3146.             d.literal_suffix,
  3147.             e.create_params,
  3148.             d.nullable,
  3149.             d.case_sensitive,
  3150.             d.searchable,
  3151.             d.unsigned_attribute,
  3152.             d.money,
  3153.             d.auto_increment,
  3154.             t.name
  3155.         from master.dbo.spt_datatype_info d, master.dbo.spt_datatype_info_ext e, systypes t
  3156.         where
  3157.             d.ss_dtype = t.type
  3158.             and t.usertype *= e.user_type
  3159.             and (t.usertype in (80,18)            /* SYSNAME or TIMESTAMP */
  3160.                 or t.usertype >= 100)            /* User defined types */
  3161.             and t.type not in (111,109,38,110)    /* get rid of nullable types */
  3162.         order by d.data_type, type_name
  3163.     else
  3164.         select    /* Real SQL Server data types */
  3165.             type_name = t.name,
  3166.             d.data_type,
  3167.             precision = isnull(d.data_precision, convert(int,t.length)),
  3168.             d.literal_prefix,
  3169.             d.literal_suffix,
  3170.             e.create_params,
  3171.             d.nullable,
  3172.             d.case_sensitive,
  3173.             d.searchable,
  3174.             d.unsigned_attribute,
  3175.             d.money,
  3176.             d.auto_increment,
  3177.             d.local_type_name
  3178.         from master.dbo.spt_datatype_info d, master.dbo.spt_datatype_info_ext e, systypes t
  3179.         where
  3180.             data_type = @data_type
  3181.             and d.ss_dtype = t.type
  3182.             and t.usertype *= e.user_type
  3183.             and t.usertype not in (80,18)        /* No SYSNAME or TIMESTAMP */
  3184.             and t.usertype < 100                /* No user defined types */
  3185.             and t.type not in (111,109,38,110)    /* get rid of nullable types */
  3186.             and t.name != 'nchar'                /* get rid of 4.9.1 type */
  3187.             and t.name != 'nvarchar'            /* get rid of 4.9.1 type */
  3188.         UNION
  3189.         select    /* SQL Server SYSNAME, TIMESTAMP and user data types */
  3190.             type_name = t.name,
  3191.             d.data_type,
  3192.             precision = isnull(d.data_precision, convert(int,t.length)),
  3193.             d.literal_prefix,
  3194.             d.literal_suffix,
  3195.             e.create_params,
  3196.             d.nullable,
  3197.             d.case_sensitive,
  3198.             d.searchable,
  3199.             d.unsigned_attribute,
  3200.             d.money,
  3201.             d.auto_increment,
  3202.             t.name
  3203.         from master.dbo.spt_datatype_info d, master.dbo.spt_datatype_info_ext e, systypes t
  3204.         where
  3205.             data_type = @data_type
  3206.             and d.ss_dtype = t.type
  3207.             and t.usertype *= e.user_type
  3208.             and (t.usertype in (80,18)            /* SYSNAME or TIMESTAMP */
  3209.                 or t.usertype >= 100)            /* User defined types */
  3210.             and t.type not in (111,109,38,110)    /* get rid of nullable types */
  3211.         order by type_name
  3212. go
  3213.  
  3214. grant execute on sp_datatype_info to public
  3215. go
  3216.  
  3217. dump tran master with truncate_only
  3218. go
  3219.  
  3220. print "creating sp_special_columns"
  3221. go
  3222.  
  3223. dump tran master with truncate_only
  3224. go
  3225.  
  3226.  
  3227. CREATE PROCEDURE sp_special_columns (
  3228.                  @table_name        varchar(32),
  3229.                  @table_owner        varchar(32) = null,
  3230.                  @table_qualifier    varchar(32) = null,
  3231.                  @col_type            char(1) = 'R')
  3232. AS
  3233.     DECLARE @indid                int
  3234.     DECLARE @table_id            int
  3235.     DECLARE @full_table_name    char(70)
  3236.     DECLARE @msg                char(70)
  3237.  
  3238.     if @col_type not in ('R','V')
  3239.        begin
  3240.           print "Invalid value - please enter either 'R' or 'V' for col_type."
  3241.           return
  3242.        end
  3243.  
  3244.     if @table_qualifier is not null
  3245.        begin
  3246.           if db_name() != @table_qualifier
  3247.               begin    /* If qualifier doesn't match current database */
  3248.                 print "Table qualifier must be name of current database"
  3249.                 return
  3250.               end
  3251.        end
  3252.     if @table_owner is null
  3253.        begin    /* If unqualified table name */
  3254.           SELECT @full_table_name = @table_name
  3255.        end
  3256.     else
  3257.        begin    /* Qualified table name */
  3258.           SELECT @full_table_name = @table_owner + '.' + @table_name
  3259.        end
  3260.     /*    Get Object ID */
  3261.     SELECT @table_id = object_id(@full_table_name)
  3262.  
  3263.     if @col_type = 'V'
  3264.     BEGIN /* if ROWVER, just run that query */
  3265.         SELECT
  3266.             scope = convert(smallint, 0),
  3267.             column_name = c.name,
  3268.             data_type = d.data_type+convert(smallint,
  3269.                             isnull(d.aux,
  3270.                             ascii(substring("666AAA@@@CB??GG",
  3271.                             2*(d.ss_dtype%35+1)+2-8/c.length,1))
  3272.                             -60)),
  3273.             type_name = t.name,
  3274.             precision = isnull(d.data_precision, convert(int,c.length))
  3275.                             +isnull(d.aux, convert(int,
  3276.                             ascii(substring("???AAAFFFCKFOLS",
  3277.                             2*(d.ss_dtype%35+1)+2-8/c.length,1))-60)),
  3278.             length = isnull(d.length, convert(int,c.length)) +convert(int,
  3279.                             isnull(d.aux,
  3280.                             ascii(substring("AAA<BB<DDDHJSPP",
  3281.                             2*(d.ss_dtype%35+1)+2-8/c.length,
  3282.                             1))-64)),
  3283.             scale = d.numeric_scale +convert(smallint,
  3284.                             isnull(d.aux,
  3285.                             ascii(substring("<<<<<<<<<<<<<<?",
  3286.                             2*(d.ss_dtype%35+1)+2-8/c.length,
  3287.                             1))-60))
  3288.         FROM
  3289.             systypes t, syscolumns c, master.dbo.spt_datatype_info d
  3290.         WHERE
  3291.             c.id = @table_id
  3292.             AND c.type = d.ss_dtype
  3293.             AND c.usertype = 80 /*    TIMESTAMP */
  3294.             AND t.usertype = 80 /*    TIMESTAMP */
  3295.         RETURN
  3296.     END
  3297.  
  3298.     /* ROWID, now find the id of the 'best' index for this table */
  3299.  
  3300.     SELECT @indid = (
  3301.         SELECT MIN(indid)
  3302.         FROM sysindexes
  3303.         WHERE
  3304.             status&2 = 2        /*    If Unique Index */
  3305.             AND id = @table_id
  3306.             AND indid > 0)        /*    Eliminate Table Row */
  3307.  
  3308.      SELECT
  3309.         scope = convert(smallint, 0),
  3310.         column_name = INDEX_COL(@full_table_name,indid,c.colid),
  3311.         data_type = d.data_type+convert(smallint,
  3312.                         isnull(d.aux,
  3313.                         ascii(substring("666AAA@@@CB??GG",
  3314.                         2*(d.ss_dtype%35+1)+2-8/c2.length,1))
  3315.                         -60)),
  3316.         type_name = rtrim(substring(d.type_name,
  3317.                         1+isnull(d.aux,
  3318.                         ascii(substring("III<<<MMMI<<A<A",
  3319.                         2*(d.ss_dtype%35+1)+2-8/c2.length,
  3320.                         1))-60), 13)),
  3321.         precision = isnull(d.data_precision, convert(int,c2.length))
  3322.                         +isnull(d.aux, convert(int,
  3323.                         ascii(substring("???AAAFFFCKFOLS",
  3324.                         2*(d.ss_dtype%35+1)+2-8/c2.length,1))-60)),
  3325.         length = isnull(d.length, convert(int,c2.length)) +convert(int,
  3326.                         isnull(d.aux,
  3327.                         ascii(substring("AAA<BB<DDDHJSPP",
  3328.                         2*(d.ss_dtype%35+1)+2-8/c2.length,
  3329.                         1))-64)),
  3330.         scale = d.numeric_scale +convert(smallint,
  3331.                         isnull(d.aux,
  3332.                         ascii(substring("<<<<<<<<<<<<<<?",
  3333.                         2*(d.ss_dtype%35+1)+2-8/c2.length,
  3334.                         1))-60))
  3335.     FROM
  3336.         sysindexes x,
  3337.         syscolumns c,
  3338.         master.dbo.spt_datatype_info d,
  3339.         systypes t,
  3340.         syscolumns c2    /* Self-join to generate list of index columns and */
  3341.                         /* to extract datatype names */
  3342.     WHERE
  3343.         x.id = @table_id
  3344.         AND c2.name = INDEX_COL(@full_table_name,@indid,c.colid)
  3345.         AND c2.id =x.id
  3346.         AND c.id = x.id
  3347.         AND c.colid < keycnt+(x.status&16)/16
  3348.         AND x.indid = @indid
  3349.         AND c2.type = d.ss_dtype
  3350.         AND c2.usertype *= t.usertype
  3351. go
  3352.  
  3353. grant execute on sp_special_columns to public
  3354. go
  3355.  
  3356. print "creating sp_databases"
  3357. go
  3358.  
  3359. create proc sp_databases
  3360. as
  3361.     /* Use temporary table to sum up database size w/o using group by */
  3362.     create table #databases (
  3363.                   database_name varchar(32),
  3364.                   size int)
  3365.  
  3366.     /* Insert row for each database */
  3367.     insert into #databases
  3368.         select
  3369.             name,
  3370.             (select sum(size) from master.dbo.sysusages
  3371.                 where dbid = d.dbid)
  3372.         from master.dbo.sysdatabases d
  3373.  
  3374.     select
  3375.          database_name,
  3376.          database_size = size*2,    /* Convert from 2048 byte pages to K */
  3377.          remarks = convert(varchar(254),null)    /* Remarks are NULL */
  3378.     from #databases
  3379.     order by database_name
  3380. go
  3381.  
  3382. grant execute on sp_databases to public
  3383. go
  3384.  
  3385. dump tran master with truncate_only
  3386. go
  3387.  
  3388. /*******************************************************************************/
  3389. /* This portion returns everything back to normal                   */
  3390. /*******************************************************************************/
  3391. use master
  3392. go
  3393.  
  3394. if exists (select * from sysobjects
  3395.             where name = 'sp_configure' and sysstat & 7 = 4)
  3396. begin
  3397.     execute sp_configure 'update',0
  3398. end
  3399. reconfigure with override
  3400. go
  3401.  
  3402. checkpoint
  3403. go
  3404.