Syntax error in recursive cte

  • Here is code tsql code using recursion within a cte that producing a syntax error marked *** below!

    DECLARE

    @dt1 Date = '2015-12-01',

    @dt2 Date = '2015-12-30',

    @SDate Date = '2015-12-01',

    @EDate Date = '2015-12-30',

    @ClientID int = 2528;

    WITH ClientAppts AS

    (

    SELECT A.ApptDate, A.ApptTime, --A.Approved +' '+

    S.FirstName + ', ' + S.Surname +' '+CAST(A.Duration AS Varchar(10)) +'mins' AS Details

    FROM TblAppointments A

    LEFT JOIN TblStaff S

    ON A.StaffID = S.PersonnelNo

    WHERE A.ClientID = @ClientID

    AND A.ApptDate BETWEEN @SDate and @EDate

    ),

    DailyDates(dates) AS

    (

    SELECT [Dates]= @dt1

    UNION ALL

    SELECT dateadd(dd,1,[dates])

    FROM DailyDates

    WHERE dateadd(dd,1,[dates]) <= @dt2)

    SELECT CONVERT(date,[dates]) AS Dates

    FROM DailyDates

    OPTION (maxrecursion 0)

    ), *** Error Here***

    SELECT A.Dates , C.ApptDate, C.ApptTime, C.Details

    FROM DailyDates A

    LEFT JOIN ClientAppts C

    ON A.dates = C.ApptDate

    END

    I have highlighted the line with the error but cant for th elife of me see what wrong all the examples I see on line match this format!

    Any ideas?

  • That END statement will cause an error if there is not a BEGIN statement somewhere before it within the same transaction. I'll just assume, however, that you have a BEGIN statement...

    I think the problem with your query is that the OPTION (maxrecursion 0) should be on the outside of your recursive CTE (right after your final LEFT JOIN statement.

    "I cant stress enough the importance of switching from a sequential files mindset to set-based thinking. After you make the switch, you can spend your time tuning and optimizing your queries instead of maintaining lengthy, poor-performing code."

    -- Itzik Ben-Gan 2001

  • The following code should be removed for it to work.

    SELECT CONVERT(date,[dates]) AS Dates

    FROM DailyDates

    OPTION (maxrecursion 0)

    ),

    However, there's a major problem. You have a rCTE that counts which can become a huge problem. Change the code into this:

    WITH

    E(n) AS(

    SELECT n FROM (VALUES(0),(0),(0),(0),(0),(0),(0),(0),(0),(0))E(n)

    ),

    E2(n) AS(

    SELECT a.n FROM E a, E b

    ),

    E4(n) AS(

    SELECT a.n FROM E2 a, E2 b

    ),

    DailyDates(Dates) AS(

    SELECT TOP (DATEDIFF( dd, @dt1, @dt2) + 1)

    DATEADD( dd, ROW_NUMBER() OVER(ORDER BY (SELECT NULL)) - 1, @dt1)

    FROM E4

    ),

    ClientAppts( ApptDate, ApptTime, Details) AS

    (

    SELECT A.ApptDate,

    A.ApptTime, --A.Approved +' '+

    S.FirstName + ', ' + S.Surname +' '+CAST(A.Duration AS Varchar(10)) +'mins' AS Details

    FROM TblAppointments A

    LEFT JOIN TblStaff S ON A.StaffID = S.PersonnelNo

    WHERE A.ClientID = @ClientID

    AND A.ApptDate BETWEEN @SDate and @EDate

    )

    SELECT A.Dates , C.ApptDate, C.ApptTime, C.Details

    FROM DailyDates A

    LEFT JOIN ClientAppts C

    ON A.dates = C.ApptDate;

    I changed the order of your CTEs just to keep the data close to the last SELECT.

    Luis C.
    General Disclaimer:
    Are you seriously taking the advice and code from someone from the internet without testing it? Do you at least understand it? Or can it easily kill your server?

    How to post data/code on a forum to get the best help: Option 1 / Option 2
  • Luis Cazares (12/30/2015)


    The following code should be removed for it to work.

    SELECT CONVERT(date,[dates]) AS Dates

    FROM DailyDates

    OPTION (maxrecursion 0)

    ),

    However, there's a major problem. You have a rCTE that counts which can become a huge problem. Change the code into this:

    WITH

    E(n) AS(

    SELECT n FROM (VALUES(0),(0),(0),(0),(0),(0),(0),(0),(0),(0))E(n)

    ),

    E2(n) AS(

    SELECT a.n FROM E a, E b

    ),

    E4(n) AS(

    SELECT a.n FROM E2 a, E2 b

    ),

    DailyDates(Dates) AS(

    SELECT TOP (DATEDIFF( dd, @dt1, @dt2) + 1)

    DATEADD( dd, ROW_NUMBER() OVER(ORDER BY (SELECT NULL)) - 1, @dt1)

    FROM E4

    ),

    ClientAppts( ApptDate, ApptTime, Details) AS

    (

    SELECT A.ApptDate,

    A.ApptTime, --A.Approved +' '+

    S.FirstName + ', ' + S.Surname +' '+CAST(A.Duration AS Varchar(10)) +'mins' AS Details

    FROM TblAppointments A

    LEFT JOIN TblStaff S ON A.StaffID = S.PersonnelNo

    WHERE A.ClientID = @ClientID

    AND A.ApptDate BETWEEN @SDate and @EDate

    )

    SELECT A.Dates , C.ApptDate, C.ApptTime, C.Details

    FROM DailyDates A

    LEFT JOIN ClientAppts C

    ON A.dates = C.ApptDate;

    I changed the order of your CTEs just to keep the data close to the last SELECT.

    I wanted to come back and mention this earlier today but we were having network issues.

    In addition to what Luis recommended I would add that this is where a properly indexed calendar table (date dimension table in the BI world) would come in handy.

    "I cant stress enough the importance of switching from a sequential files mindset to set-based thinking. After you make the switch, you can spend your time tuning and optimizing your queries instead of maintaining lengthy, poor-performing code."

    -- Itzik Ben-Gan 2001

  • Hi Luis,

    Thanks for your prompt reply.

    I know your code works as I have tested it out before. I know it creates a cte of dates but would you mind explaining how it works as I cant seem to grasp it at all. why do we need nested cte's?

    E(n) AS(

    SELECT n FROM (VALUES(0),(0),(0),(0),(0),(0),(0),(0),(0),(0))E(n)

    ),

    E2(n) AS(

    SELECT a.n FROM E a, E b

    ),

    E4(n) AS(

    SELECT a.n FROM E2 a, E2 b

    ),

    DailyDates(Dates) AS(

    SELECT TOP (DATEDIFF( dd, @dt1, @dt2) + 1)

    DATEADD( dd, ROW_NUMBER() OVER(ORDER BY (SELECT NULL)) - 1, @dt1)

    FROM E4

    Also someone suggested creating a simple single column table of dates which on the face of it has the advantage of being easier to handle and maintain, I would only need 6 months of dates at any one time?

    kind regards

  • Tallboy (12/30/2015)


    Hi Luis,

    Thanks for your prompt reply.

    I know your code works as I have tested it out before. I know it creates a cte of dates but would you mind explaining how it works as I cant seem to grasp it at all. why do we need nested cte's?

    This is a common creation of a cteTally which won't need to read from disk any information, based on the original model of Itzik Ben-Gan.

    It creates sets of rows that are exponentially growing on each CTE which then generates the numbers or dates in this case.

    Here's the commented version:

    WITH

    E(n) AS(

    --We create 10 rows

    SELECT n FROM (VALUES(0),(0),(0),(0),(0),(0),(0),(0),(0),(0))E(n)

    ),

    E2(n) AS(

    --With a cross join we create 10 x 10 = 100 rows

    SELECT a.n FROM E a, E b

    ),

    E4(n) AS(

    --With a cross join we create 100 x 100 = 10,000 rows

    SELECT a.n FROM E2 a, E2 b

    ),

    DailyDates(Dates) AS(

    SELECT TOP (DATEDIFF( dd, @dt1, @dt2) + 1) --We limit the rows to what we need

    DATEADD( dd, ROW_NUMBER() OVER(ORDER BY (SELECT NULL)) - 1, @dt1) --We add n-1 days to the initial date

    FROM E4

    )

    SELECT A.Dates

    FROM DailyDates A ;

    You don't actually need to create all of them and can simplify the code.

    WITH

    DailyDates(Dates) AS(

    SELECT TOP (DATEDIFF( dd, @dt1, @dt2) + 1) --We limit the rows to what we need

    DATEADD( dd, ROW_NUMBER() OVER(ORDER BY (SELECT NULL)) - 1, @dt1) --We add n-1 days to the initial date

    FROM (VALUES(0),(0),(0),(0),(0),(0),(0),(0),(0),(0))E(n),

    (VALUES(0),(0),(0),(0),(0),(0),(0),(0),(0),(0))E2(n),

    (VALUES(0),(0),(0),(0),(0),(0),(0),(0),(0),(0))E3(n) -- 10 x 10 x 10 = 1000 rows

    )

    SELECT A.Dates

    FROM DailyDates A ;

    Hope this will clarify how the code works.

    Luis C.
    General Disclaimer:
    Are you seriously taking the advice and code from someone from the internet without testing it? Do you at least understand it? Or can it easily kill your server?

    How to post data/code on a forum to get the best help: Option 1 / Option 2
  • Good catch, Luis.

    Tallboy, if you aren't familiar with tally tables and how they can be used, see the article at http://www.sqlservercentral.com/articles/T-SQL/62867/ for explanation and examples.

  • Hi Guys, I have the above code running well now but have run into another problem with a pivot table

    ,AllData AS

    (

    SELECT A.Dates , C.ApptTime, C.Details

    FROM DailyDates A

    LEFT JOIN ClientAppts C

    ON A.dates = C.ApptDate

    )

    SELECT *

    FROM AllData

    PIVOT

    (

    MAX(Details)

    FOR Dates IN (SELECT Dates FROM DailyDates)

    )

    AS p

    SQL won't let me use a SELECT clause inside the PIVOT !

    Help!

  • HERE is my latest attempt to solve the problem...

    while it produce a correct list of dates on my @cols string. I need the dates to be wrapped with singel quotes...i.e. '2015-12-20',

    AllData AS

    (

    SELECT A.Dates , C.ApptTime, C.Details --C.ApptDate,

    FROM DailyDates A

    LEFT JOIN ClientAppts C

    ON A.dates = C.ApptDate

    )

    SELECT @cols = ISNULL(@cols +',', '') + CAST(Dates AS VArchar) FROM DailyDates

    SELECT @SQLString = N'SELECT A.Dates , A.ApptTime, A.Details

    FROM AllData A

    PIVOT

    (

    MAX(A.Details)

    FOR A.Dates IN ('+ @cols +')

    )

    AS p;'

    EXEC sp_executesql @SQLString

    --SELECT @cols

    END

    Any advise appreciated!

  • You can't create a dynamic pivot like that. You need to use dynamic SQL.

    In my case, I prefer to use cross tabs instead of pivots. These is because they allow more flexibility and can give better performance when more than one pivot is needed in a single query.

    For further explanation on this, check the following articles:

    http://www.sqlservercentral.com/articles/T-SQL/63681/

    http://www.sqlservercentral.com/articles/Crosstab/65048/

    Here's an example of what you need according to my crystal ball:

    CREATE TABLE TblAppointments(

    ApptDate date,

    ApptTime time,

    Duration int,

    StaffID int,

    ClientID int

    );

    INSERT INTO TblAppointments

    VALUES

    ( '20151205', '13:00', 30, 1, 2528),

    ( '20151206', '11:00', 60, 1, 2528),

    ( '20151207', '15:00', 15, 1, 2528),

    ( '20151208', '18:00', 10, 1, 2528),

    ( '20151209', '12:00', 90, 1, 2528),

    ( '20151210', '11:00', 120, 1, 2528);

    CREATE TABLE TblStaff(

    PersonnelNo int,

    FirstName varchar(50),

    Surname varchar(50)

    );

    INSERT INTO TblStaff VALUES(1, 'Luis', 'Cazares');

    DECLARE

    @dt1 Date = '2015-12-01',

    @dt2 Date = '2015-12-30',

    @SDate Date = '2015-12-01',

    @EDate Date = '2015-12-30',

    @ClientID int = 2528;

    DECLARE @sql nvarchar(max),

    @Header nvarchar(max),

    @Columns nvarchar(max),

    @Footer nvarchar(max);

    --Header

    SELECT @Header = 'WITH ClientAppts AS

    (

    SELECT A.ApptDate, A.ApptTime,

    S.FirstName + '', '' + S.Surname + '' '' + CAST(A.Duration AS Varchar(10)) + '' mins'' AS Details

    FROM TblAppointments A

    LEFT JOIN TblStaff S ON A.StaffID = S.PersonnelNo

    WHERE A.ClientID = @ClientID

    AND A.ApptDate BETWEEN @SDate and @EDate

    ),

    Times(Hours) AS(

    SELECT TOP (24) --We limit the rows to what we need

    CAST( DATEADD( HH, ROW_NUMBER() OVER(ORDER BY (SELECT NULL)) - 1, 0) AS time)

    FROM (VALUES(0),(0),(0),(0),(0))E(n),

    (VALUES(0),(0),(0),(0),(0))E2(n) -- 5 x 5 = 25 rows

    )

    SELECT CAST( A.Hours AS CHAR(5)) AS ApptTime';

    WITH DailyDates(Dates) AS(

    SELECT TOP (DATEDIFF( dd, @dt1, @dt2) + 1) --We limit the rows to what we need

    DATEADD( dd, ROW_NUMBER() OVER(ORDER BY (SELECT NULL)) - 1, @dt1) --We add n-1 days to the initial date

    FROM (VALUES(0),(0),(0),(0),(0),(0),(0),(0),(0),(0))E(n),

    (VALUES(0),(0),(0),(0),(0),(0),(0),(0),(0),(0))E2(n),

    (VALUES(0),(0),(0),(0),(0),(0),(0),(0),(0),(0))E3(n) -- 10 x 10 x 10 = 1000 rows

    )

    SELECT @Columns = (SELECT CHAR(13) + CHAR(9) + ',MAX( CASE WHEN C.ApptDate = ' + QUOTENAME(CONVERT( char(8), Dates, 112), '''')

    + ' THEN C.Details END) AS ' + QUOTENAME(CONVERT( char(10), Dates, 120))

    FROM DailyDates

    FOR XML PATH(''), TYPE).value('.', 'nvarchar(max)');

    SELECT @Footer = 'FROM Times A

    LEFT JOIN ClientAppts C ON A.Hours = C.ApptTime

    GROUP BY A.Hours;';

    SET @sql = @Header + @Columns + @Footer;

    PRINT @sql;

    EXEC sp_executesql @sql,

    N'@SDate Date, @EDate Date, @ClientID int',

    @SDate,

    @EDate,

    @ClientID;

    GO

    DROP TABLE TblAppointments;

    DROP TABLE TblStaff;

    Note that I had to create the sample data to be able to test. You should provide your sample data this way for future posts.

    Luis C.
    General Disclaimer:
    Are you seriously taking the advice and code from someone from the internet without testing it? Do you at least understand it? Or can it easily kill your server?

    How to post data/code on a forum to get the best help: Option 1 / Option 2

Viewing 10 posts - 1 through 9 (of 9 total)

You must be logged in to reply to this topic. Login to reply