Insert using table unput

  • I have a table that I need to run off a different server\db and looking for the most efficient way to call from an SP. I need to execute.

    the field(sqlToExec) as it's loaded with 1000's of Insert statements like below to update this same table on a different server and DB.

    If something error could we catch in an errorlog file or, table and continue processing, and should the Inserts be Batched by ID say 30,000 at a time.

     

    Sample data:

    INSERT INTO EngData (PartNbr,CostKey,PlantCd,Operation,Machine,RunHrs,SetHrs,ScrapRate,Tasking,Efficiency,Speed,NomOD,Weight,AvgOrder,Com1Wt,RecovCode,RecovDescr,RecovRate,SpecDesc1)

    VALUES('814000000','Def','Tst','COLORING','FCOLOR',0.00193493,null,0.008,2.66,47,2100,0.255,0.05004,0,0,'05','No Copper',0,'1-RWP NAT')

    Any help much appreciated.

    CREATE TABLE [dbo].[crtInsert](
    [id] [int] IDENTITY(1,1) NOT NULL,
    [sqlToExec] [nvarchar](4000) NULL,
    [timeAdded] [datetime] NULL
    ) ON [PRIMARY]
    GO

    ALTER TABLE [dbo].[crtInsert] ADD DEFAULT (getdate()) FOR [timeAdded]
    GO
  • Looks like you are going to need to use dynamic SQL and you are going to have a bad time with that I fear. Why? because what happens if someone puts in something like "DROP DATABASE PRODUCTION" in the "sqlToExec" statement?

    BUT if this is what you HAVE to do and you don't care about the risks, then look up dynamic SQL and I expect you'll also need to use a cursor to loop through the table.

    BUT I do strongly advise against this approach as it is high risk.

    The above is all just my opinion on what you should do. 
    As with all advice you find on a random internet forum - you shouldn't blindly follow it.  Always test on a test server to see if there is negative side effects before making changes to live!
    I recommend you NEVER run "random code" you found online on any system you care about UNLESS you understand and can verify the code OR you don't care if the code trashes your system.

  • The query that feeds this table is very specific (Only Insert) in what gets added to the sqltoexec can it be done without using a cursor?

    Thanks for your reply.

  • I tried this but it didn't delete record

    DECLARE @Counter INT = 1;
    DECLARE @MaxCount INT = 1; -- Set this to the number of times you want to run the insert
    DECLARE @SQLString NVARCHAR(MAX);
    DECLARE @ParmDefinition NVARCHAR(MAX);

    WHILE @Counter <= @MaxCount
    BEGIN
    -- Construct the SQL command
    SET @SQLString = N'select sqlToExec from sv1.cb.dbo.crtDelete where ID =1';
    -- Execute the SQL command
    EXECUTE sp_executesql @SQLString
    -- Increment the counter
    SET @Counter = @Counter + 1;
    END
  • Why would it? Your dynamic SQL starts with a select, so the quiet getting executed is a select. You would want something more along the lines of

    Select @SQLString = sqlToExec
    From sv1.cb....

    Sorry for the bad code example... I am doing this from my phone and it is not letting me edit the code but you get the Idea (hopefully).

    Also, just an FYI, a while loop in SQL is essentially a cursor.

    The above is all just my opinion on what you should do. 
    As with all advice you find on a random internet forum - you shouldn't blindly follow it.  Always test on a test server to see if there is negative side effects before making changes to live!
    I recommend you NEVER run "random code" you found online on any system you care about UNLESS you understand and can verify the code OR you don't care if the code trashes your system.

  • I must have something wrong as it's error'ing

    This is what a single sqlToExec where ID=1  looks like

    delete from cstSums WHERE PartNbr='81K000000' AND PlantCd='CORP' AND CostKey='Standard'

     

    Thanks for reply.

  • You day it is erroring, what is the error?

    Also try printing you dynamic SQL before you run it so you can verify it is what you expect.

    The above is all just my opinion on what you should do. 
    As with all advice you find on a random internet forum - you shouldn't blindly follow it.  Always test on a test server to see if there is negative side effects before making changes to live!
    I recommend you NEVER run "random code" you found online on any system you care about UNLESS you understand and can verify the code OR you don't care if the code trashes your system.

  • DECLARE @Counter INT = 1;
    DECLARE @MaxCount INT = 1; -- Set this to the number of times you want to run the insert
    DECLARE @SQLString NVARCHAR(MAX);
    DECLARE @ParmDefinition NVARCHAR(MAX);

    WHILE @Counter <= @MaxCount
    BEGIN
    -- Construct the SQL command
    Select @SQLString = N'sqlToExec from cb.dbo.crtDelete where ID =1';
    -- Execute the SQL command

    Select @SQLString
    EXECUTE sp_executesql @SQLString
    -- Increment the counter
    SET @Counter = @Counter + 1;
    END

    The select of @sqlstring is:

    sqlToExec from cb.dbo.crtDelete where ID =1

    This is what the sqltoexec should be running.

    delete from cstSums WHERE PartNbr='81K000000' AND PlantCd='CORP' AND CostKey='Standard'

  • Yes, that is what I was saying. Your command for setting the SQL string is wrong.

    Again, I STRONGLY encourage you to re-evaluate your solution as I am not sure if you and your team can support the solution. I'm not trying to question your SQL ability, but I want to make sure you can support this going forward. What will you do if/when this process fails? This forum can help, but it may take days to help you or we may not be able to help (we have no access to your systems or data).

    BUT if you are 100% confident this is the solution you want to use and are 100% confident supporting it, the solution to your problem is:

    DECLARE @Counter INT = 1;
    DECLARE @MaxCount INT = 1; -- Set this to the number of times you want to run the insert
    DECLARE @SQLString NVARCHAR(MAX);
    DECLARE @ParmDefinition NVARCHAR(MAX);

    WHILE @Counter <= @MaxCount
    BEGIN
    -- Construct the SQL command
    Select @SQLString = CAST(sqlToExec AS NVARCHAR(MAX)) from cb.dbo.crtDelete where ID = 1;
    -- Execute the SQL command

    PRINT @SQLString
    -- EXECUTE sp_executesql @SQLString
    -- Increment the counter
    SET @Counter = @Counter + 1;
    END

    Things I changed:

    1- the SELECT for the @SQLString now gets the correct value back

    2- I changed the SELECT to a PRINT so you can review the results and it will handle larger data sets better

    3- I commented out the EXECUTE as you should ALWAYS validate that your SQL is valid prior to running dynamic SQL.

    Another thing to note - your sqltoexec sample is a bit risky because you are NOT including the schema in the database name. This is risky because you CAN have 2 tables with the same name but different schemas. IF user A is expecting mySchema.cstSums to have data deleted as that is the table they usually access BUT your job runs as a user with dbo as the default schema AND you have a table called "cstSums" in the dbo schema as well, it'll delete from the dbo schema. IF there is no such table, you will get an error. You should ALWAYS (100% of the time) include the schema when you work with a table. Including the database should ONLY be done IF you are running a cross-database query. Otherwise it runs the risk of you running the query against the wrong database.

    The above is all just my opinion on what you should do. 
    As with all advice you find on a random internet forum - you shouldn't blindly follow it.  Always test on a test server to see if there is negative side effects before making changes to live!
    I recommend you NEVER run "random code" you found online on any system you care about UNLESS you understand and can verify the code OR you don't care if the code trashes your system.

  • Mr. Brian Gale wrote:

    Including the database should ONLY be done IF you are running a cross-database query.

    Otherwise it runs the risk of you running the query against the wrong database.

    … Or of failing, if the database (or a copy of it) is ever renamed.

    … Or of thoroughly confusing Visual Studio from within a database project.

    The absence of evidence is not evidence of absence.
    Martin Rees

    You can lead a horse to water, but a pencil must be lead.
    Stan Laurel

  • Thanks for replies and alarms of this type of approach -- defiantly something that has been discussed.

    With the approach above how it will scale when removing the where ID =1 .. since this was just for testing... There are going to be 1000,s of requests in this queue(sqltoexec) query... Can it be batched... how about catching failures can that be logged?

    Many thanks again

     

     

  • You can do this however you like. It's your system. You want it batched? then write your code to batch it. You want it to capture failures, toss in some error handling. As for scaling, it will not scale well. Cursors and loops are HUGE performance sucks which is why database people avoid them as much as possible.

    If you want this to scale, you need to change the approach and likely the application.

    One big way to get it to scale better - don't throw your SQL into a table to be run at a future date. Have the application run the SQL directly. OR, better yet, have a stored procedure that takes parameters of what is to be deleted (or inserted or selected or updated) and have the application call the stored procedure. That way your query can hit the plan cache rather than being a bunch of ad-hoc queries.

    Also, with your approach, you would want to do "something" with a query after it has been run. What I mean is right now, lets say you have 10 rows in that table and you run your process to execute those. You will loop through 10 times. Now, you run your process again immediately after it completes - still running 10 times. Now users add 90 rows and you run it again, do you want it to run 100 times (old rows + new rows) or 90 times (just new rows)?

    The above is all just my opinion on what you should do. 
    As with all advice you find on a random internet forum - you shouldn't blindly follow it.  Always test on a test server to see if there is negative side effects before making changes to live!
    I recommend you NEVER run "random code" you found online on any system you care about UNLESS you understand and can verify the code OR you don't care if the code trashes your system.

  • The table will be cleared out before this new process runs, but I do need to process by ID..

    Could it multi process the table?

    Thanks

  • Bruin wrote:

    The table will be cleared out before this new process runs, but I do need to process by ID..

    Could it multi process the table?

    Thanks

    I would not recommend "multi-processing" the table (if I understand what you mean by handling multiple ID's concurrently). The reason being potential for a missed ID or duplicate ID's being run (low risk if you code it well) and blocking (medium risk if you have multiple scripts deleting from the same table, but it'll just pause one processing until the other completes. Worst case though, you'll hit a deadlock and something will get rolled back that you will need to re-process. I would NOT recommend "multi processing".

    BUT if you want to, nothing me (or anyone else on this forum) can do to stop you! It's your code :). Just set things up so you can control the start and stop ID and run multiple queries in multiple SSMS windows or with multiple jobs.

    Still not something I'd do though. IF I had to do something like this, I'd do it as a data export to Excel where I can review all the commands and put some sanity checking in there. Then if all scripts pass my sanity checks, I can copy-paste them to SSMS and run them and capture all errors. I would not have an automated system running arbitrary SQL which could be manipulated to be malicious and break my system. What if someone forgets a WHERE clause for example? You OK with the script just deleting ALL records from a table? Can you recover in that scenario? Or what if a malicious person puts a drop database in there? can you recover? or they sneak a CREATE LOGIN and grant it sysadmin access... soooooo much can go wrong with automated dynamic SQL. At least if I dump it to Excel, I have a record of what was run and I can verify before execution. I can run it through all my sanity checks (does LOGIN exist in the script? does DROP exist in the script? If DELETE exists, is there a WHERE? etc). Doesn't help recover from errors, but helps me make sure the scripts are accurate and complete prior to running them.

    The above is all just my opinion on what you should do. 
    As with all advice you find on a random internet forum - you shouldn't blindly follow it.  Always test on a test server to see if there is negative side effects before making changes to live!
    I recommend you NEVER run "random code" you found online on any system you care about UNLESS you understand and can verify the code OR you don't care if the code trashes your system.

  • Thanks Brian for all of your replies. and suggestions can you look at this code and tell me if this would handle processing the table with a batchsize of 50,000 and process my entire input table?

    I ran it a few times with Print statement but doesn't look like I'm doing the entire table and doesn't seem to  remove records ..

    Thanks again

    DECLARE @BatchSize INT = 50000;
    DECLARE @IdControl INT = 1;
    DECLARE @Results INT;
    DECLARE @SQLString NVARCHAR(MAX);
    DECLARE @ParmDefinition NVARCHAR(MAX);

    WHILE (@Results IS NULL OR @Results > 0)

    BEGIN


    -- Insert records in a batch
    Select @SQLString = CAST(sqlToExec AS NVARCHAR(MAX)) from CB_Staging.dbo.crtDelete where
    ID > @IdControl AND ID <= (@IdControl + @BatchSize) order by ID;

    Print @SQLString
    -- Execute the SQL command
    Begin Try
    EXECUTE sp_executesql @SQLString
    END TRY
    BEGIN CATCH
    -- Error handling code
    SELECT
    ERROR_NUMBER() AS ErrorNumber,
    ERROR_SEVERITY() AS ErrorSeverity,
    ERROR_STATE() AS ErrorState,
    ERROR_PROCEDURE() AS ErrorProcedure,
    ERROR_LINE() AS ErrorLine,
    ERROR_MESSAGE() AS ErrorMessage;
    -- You can log the error details or take other actions as needed
    END CATCH
    -- Update the @IdControl for the next batch
    SET @Results = @@ROWCOUNT
    SET @IdControl = @IdControl + @BatchSize

    END

     

Viewing 15 posts - 1 through 14 (of 14 total)

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