SQL Server stores information about all objects and their properties as metadata that can be accessed through system views. In addition, some of the system views hold interesting nuances that can help to better understand how a DBMS works.
To see the system view body, just as for any other script object, the OBJECT_DEFINITION function is used:
However, OBJECT_DEFINITION, as well as its analogue sp_helptext, has a significant disadvantage; it does not allow the return script description for a table object.
When executing sp_helptext, we will get the following error:
Msg 15197, Level 16, State 1, Procedure sp_helptext, Line 107
There is no text for object 'dbo.Table1'.
Under the same conditions, a system function OBJECT_DEFINITION returns NULL.
Fetching from sys.sql_modules will also not solve the problem, since the same old OBJECT_DEFINITION function call is used inside this system view:
Such behavior is rather disappointing. Sometimes it is neecessary to retrieve a script description of a table for some scripts. Well, let’s look at system views and create an OBJECT_DEFINITION function analogue for working with table objects.
To start, let's create a test table, in order for the process of script writing to be more clear:
And proceed to the first step, getting a list of columns and their properties.
Essentially, the list of columns can be obtained by simply referencing one of the several system views. Thus, it is important to fetch from the simplest system views, in order for the query execution time to be minimal.
Here are a few examples along with their execution plans made in dbForge Studio for SQL Server:
The presented plans show that the #1 and #2 approaches contain excessive amount of connections that will increase the query execution time, while the #3 approach leads to the complete scan of the index, making it the least efficient of all.
In terms of performance, the #4 approach remains the most attractive to me.
However, data contained in sys.columns (as well as in INFORMATION_SCHEMA.COLUMNS) is not enough to completely describe the table structure. This forces joins to other system views to be established as in the following:
Accordingly, the execution plan will look not so optimistic, as before. Note that the column list is even read out 3 times:
Have a look inside sys.default_constraints:
There is an OBJECT_DEFINITION call inside the system view. So, to retrieve the description of the default constraint, we don’t need to establish joining.
OBJECT_DEFINITION is still used in sys.computed_columns:
We seem to have already avoided 2 joins. The case with
sys.identity_columns is more curious:
To retrieve information about IDENTITY properties, an undocumented property IDENTITYPROPERTY is used. After a check, its unchanging behavior on SQL Server 2005 and higher was ascertained.
As a result of calling these functions directly, the column list obtaining query becomes significantly simplified:
And the execution plan becomes more efficient:
Finally, instead of joining to sys.schemas, the SCHEMA_NAME system function can be called, that triggers much more faster than JOIN. This is true, provided that the number of schemes does not exceed the number of user objects. And since such a situation is unlikely, it can be neglected.
Next, get a list of columns included in the primary key. The most obvious approach is to use sys.key_constraints:
In most cases,
PRIMARY KEY is a clustered index and the
Unique constraint.
At the metadata level, SQL Server sets index_id to 1 for all clustered indexes, so we can make a selection from sys.indexes filtering by is_primary_key = 1 or by index_id = 1 (not recommended).
Additionally, to avoid joining to sys.columns, the COL_NAME system function can be used:
Now combine the obtained queries into one query to get the following final query:
Which, when executed, will generate the following script for the test table:
As you can see, the topic is too broad and it is not limited to a column list and primary key.
That's why generation of indexes, foreign keys, and other statements are planned to be revealed in the next part of this topic.