NetTalk Central

Author Topic: SQL Sleeping Connections  (Read 22866 times)

trent

  • Full Member
  • ***
  • Posts: 204
    • View Profile
    • Email
Re: SQL Sleeping Connections
« Reply #15 on: August 18, 2013, 02:17:07 PM »
What about when you are hand-coding a loop through a table? Would you set the timeout on and off for each record accessed or just once for the entire loop?

kevin plummer

  • Hero Member
  • *****
  • Posts: 1195
    • View Profile
    • Production Accounting and Software Payroll
Re: SQL Sleeping Connections
« Reply #16 on: August 18, 2013, 06:07:53 PM »
I do it for every file access. eg

      TAGGED{Prop:SQL} = 'Set Lock_Timeout 10000'      !wait 10 seconds
      TAGGED{Prop:SQL} = 'Select * from TAGGED Where TAG_SessionId = <39>' & clip(p_web.SessionId) & '<39> and TAG_Tagged = 1 and TAG_Module = <39>POT<39>'
      TAGGED{Prop:SQL} = 'Set Lock_Timeout -1'      !
      Loop
        Next(Tagged)
        If Errorcode()
          Break
       End
       If Tag:Tagged = 1
         Tag:Tagged = 0
         TAGGED{Prop:SQL} = 'Set Lock_Timeout 10000'      !wait 10 seconds
          Access:TAGGED.Update()
         TAGGED{Prop:SQL} = 'Set Lock_Timeout -1'      !
       End
    End

I think this is better than holding the timeout until the end of the loop. It can be problematic if you call a sub routine and you set the timeout again before you set it off or if you forget to set the timeout off. This is just my experience...

Kev

trent

  • Full Member
  • ***
  • Posts: 204
    • View Profile
    • Email
Re: SQL Sleeping Connections
« Reply #17 on: August 18, 2013, 08:01:41 PM »
OK but is it needed for each 'next()'? i.e.

access:UserLicence.open
    access:UserLicence.usefile()
    clear(ULI:Record)
    ULI:UserID = p_web.GSV('UserID')
    set(ULI:KeyUserModuleLevel,ULI:KeyUserModuleLevel)
    LOOP
      p_web.SetSqlTimeout(UserLicence, net:on) <<<<<<<<here?
      if access:UserLicence.next() then break.
      if ULI:UserID not = p_web.GSV('UserID') then break.
      p_web.SSV(p_web.GSV('UserID') &'-'& clip(ULI:ModuleName),ULI:LicenceLevel)
      p_web.SetSqlTimeout(UserLicence, net:Off) <<<<<<<<and here?
    end!loop
    access:UserLicence.close

Also would you go so far as to including it for the 'open', 'usefile()' and 'close'?

kevin plummer

  • Hero Member
  • *****
  • Posts: 1195
    • View Profile
    • Production Accounting and Software Payroll
Re: SQL Sleeping Connections
« Reply #18 on: August 20, 2013, 06:42:35 AM »
if using prop:sql, no because it's in the buffer. not sure about abc access but I "guess" it just grabs 1 record at a time and in that case yes.

however in your example there is a problem where you break out of the loop without setting the timeout off. So I would either set the timeout on/off outside the LOOP or rework you next/break statement so you set off before exiting the loop.

If using ABC I think I just set it on/off outside the loop - seemed to work...

trent

  • Full Member
  • ***
  • Posts: 204
    • View Profile
    • Email
Re: SQL Sleeping Connections
« Reply #19 on: August 20, 2013, 12:42:52 PM »
Thanks for clearing that up Kevin. I think I'll set it for each record in an ABC loop just in case. Having a specific time limit on the entire loop wouldn't work for some of our queries.

I wonder if there is a more convenient way to do this... will have to research a bit further.

Regards,
Trent

trent

  • Full Member
  • ***
  • Posts: 204
    • View Profile
    • Email
Re: SQL Sleeping Connections
« Reply #20 on: January 21, 2014, 05:26:25 PM »
Hi All,

I've set all handcoded SQL statements to have a timeout now BUT our WebServer and main application are still getting 'blocked' by these sleeping connections to the point that the WebServer has to be restarted at least 2 or three times a day to allow our main application to work correctly (Win32 browses and update form windows just freeze when a user tries to open the window). This looks really bad for us and shouldn't be happening.

A quick google search suggests that disabling or even killing all active sleeping connections is a bad idea and will cause performance issues, so what else can be done to fix this issue?

Does our Win32 application need to have the SetTimeout property set on every handcoded and non-handcoded SQL statement?

Regards,
Trent

kevin plummer

  • Hero Member
  • *****
  • Posts: 1195
    • View Profile
    • Production Accounting and Software Payroll
Re: SQL Sleeping Connections
« Reply #21 on: January 21, 2014, 09:18:57 PM »
Hi Trent,

> getting 'blocked' by these sleeping connections

- sleeping connections don't block apps - just the ones that are locked

> killing all active sleeping connections is a bad idea and will cause performance issues, so what else can be done to fix this issue?

- correct. what you want is to bring up the blocked and blocking process. On the blocking process's see what SQL statement it is running. Also look at what Table is being locked. You can then kill individual connections which sometimes works without killing your app.

> Does our Win32 application need to have the SetTimeout property set on every handcoded and non-handcoded SQL statement?

- I don't believe so.

> I've set all handcoded SQL statements to have a timeout now

- if you know what table is being locked then that narrows things a lot. If you miss resetting a timeout (off) after your code then you will experience locks. My feeling is something is missing somewhere.

> WebServer has to be restarted at least 2 or three times a day to allow our main application to work correctly

- I used to have code placed in strategic places that would run some SQL statements (like a sum or count) on tables normally affected and if it came back with the usual errorcode when a table is locked I would automatically restart the app. I also used Xfiles to save all session data before the restart and then reload it when the app restarted so users could carry on without really knowing there was a problem.

trent

  • Full Member
  • ***
  • Posts: 204
    • View Profile
    • Email
Re: SQL Sleeping Connections
« Reply #22 on: January 22, 2014, 12:07:04 AM »
Hi Kevin,

>>You can then kill individual connections which sometimes works without killing your app.

Yeah I don't really want to be forever having to log into client's servers to manually kill individual connections - I like making things easy for myself not harder :)

>>My feeling is something is missing somewhere.

I'll go through the entire apps again to check. The environment is two Win32 apps using 2x separate databases (one database merges data into the second database, the 1st app can call procedures in the 2nd apps dlls). The WebServer is used for the 1st Win32 app.

>>I would automatically restart the app.

The WebServer runs on a master server. Including this functionality would mean including more 3rd party programs that can start remotely from other machines on the network (unless NetTalk can do this???).

>>I also used Xfiles to save all session data before the restart and then reload it when the app restarted so users could carry on without really knowing there was a problem.

That's pretty awesome how do you do that?

Regards,
Trent

kevin plummer

  • Hero Member
  • *****
  • Posts: 1195
    • View Profile
    • Production Accounting and Software Payroll
Re: SQL Sleeping Connections
« Reply #23 on: January 22, 2014, 02:06:45 AM »
It's all done in the web app...

Bruce gets the credit on the xfiles stuff as I needed his expertise to get it working. I log all my messages to disk which then helps me identify which tables were locked etc. Hope this helps. One app that restarted 5-6 times a day now runs for months on end. Another app my uses used to email me with SWOD (spinning wheel of Death) happened so often the app was almost unusable and caused them (and me) a lot of grief. Both fixed with the timeout stuff. I also run an app that comes with self service connectionchecker that also checks the app is running and if not restarts it.

I had this in it's own source procedure hooked off the indexpage so it regularly checked for locks.

CheckForLocks    ROUTINE
    p_web.SetSQLTimeout(ADJUST,net:on)
    DUMMY{Prop:SQL} = 'Select Count(*) as Dum_Field01 From APPROVAL'
    p_web.SetSQLTimeout(ADJUST,net:off)
    Next(DUMMY)
    If Errorcode()
      Case Errorcode()
      Of 33
        p_web._trace('Lock on APPROVAL - restart errorcode=' & errorcode() & 'error=' & error() & ' ' & clip(glo:dbowner))
        Message('Lock on APPROVAL - restart errorcode=' & errorcode() & 'error=' & error() & ' Threads=' & 'wp:NumberOfThreads')
        p_web.RequestData.Webserver._wait()
        xml1.save(p_web.RequestData.Webserver._SessionQueue,'sessionqueue.xml','table','record')
        xml1.save(p_web.RequestData.Webserver._SessionDataQueue,'sessiondataqueue.xml','table','record')
        p_web.RequestData.Webserver._release()
        Chain(Command(0))
      Of 90
        p_web._trace('APPROVAL Busy=' & errorcode() & 'error=' & error() & ' ' & clip(glo:dbowner))
        Message('APPROVAL Busy=' & errorcode() & 'error=' & error() & ' ' & glo:dbowner)
        p_web.RequestData.Webserver._wait()
        xml1.save(p_web.RequestData.Webserver._SessionQueue,'sessionqueue.xml','table','record')
        xml1.save(p_web.RequestData.Webserver._SessionDataQueue,'sessiondataqueue.xml','table','record')
        p_web.RequestData.Webserver._release()
        Chain(Command(0))
      Else
        p_web._trace('APPROVAL Error=' & errorcode() & 'error=' & error() & ' ' & clip(glo:dbowner))
        Message('APPROVAL Error=' & errorcode() & 'error=' & error() & ' Threads=' & 'wp:NumberOfThreads')
      End
    Else

    End

***********************************

This stuff goes in the webserver procedure

  GlobalErrors.SetProcedureName('WebServer')
! [Priority 800]
  Message('Web Server started: ' & LongPath())
  xml.TagCase = XF:CaseAny
  xml.load(ThisWebServer._SessionQueue,'sessionqueue.xml','table','record')
  xml.load(ThisWebServer._SessionDataQueue,'sessiondataqueue.xml','table','record')

***
ThisWindow.Kill PROCEDURE

ReturnValue          BYTE,AUTO

! Start of "WindowManager Method Data Section"
! [Priority 5000]

! End of "WindowManager Method Data Section"
  CODE
  ! Start of "WindowManager Method Executable Code Section"
  ! [Priority 50]
   ! ThisWebServer._Wait()
    xml.save(ThisWebServer._SessionQueue,'sessionqueue.xml','table','record')
    xml.save(ThisWebServer._SessionDataQueue,'sessiondataqueue.xml','table','record')
   ! ThisWebServer._Release()


****
xml.SaveCurrentFieldToXML   PROCEDURE (Long p_x,Long p_DimCounter,String p_name)
! Start of "Class Method - Data Section"
! [Priority 5000]
a  string(1)

! End of "Class Method - Data Section"
  CODE
! Start of "Class Method - Executable Code Section"
! [Priority 50]
  ! xfiles doesn't do &Strings well (yet), so we need to give it a hand here, when saving.
  if lower(p_name) = 'extvalue'
    if thisWebServer._SessionDataQueue.ExtValue &= Null
      self.currentfield &= a
    else
      self.currentfield &= thisWebServer._SessionDataQueue.ExtValue
    end
  end


******
xml.AssignField   PROCEDURE (String pString)
! Start of "Class Method - Data Section"
! [Priority 5000]

! End of "Class Method - Data Section"
  CODE
! Start of "Class Method - Executable Code Section"
! [Priority 50]
  ! the incoming queue contains the length of memory set aside for the (bigger) session values.
  ! and big values are stored in extvalue, not value. Extvalue is a pointer ( &string) so we need to
  ! give xFiles some assistance here.
  ! This code will only work if extvaluesize is encountered _before_ extvalue.

  case lower(self.CurrentTag)
  of 'extvaluesize'
    thisWebServer._SessionDataQueue.ExtValueLen = pString ! pString is a parameter to this method, not the pString data type.
    thisWebServer._SessionDataQueue.ExtValue &= new(string(thisWebServer._SessionDataQueue.ExtValueLen))
    return
  of 'extvalue'
    if pString <> ''
      thisWebServer._SessionDataQueue.ExtValue = pString
    end
    return
  end


*****






debraballenger

  • Newbie
  • *
  • Posts: 49
    • View Profile
    • Email
Re: SQL Sleeping Connections
« Reply #24 on: January 22, 2014, 07:14:31 AM »
If a sleep connection is blocking, that indicates that the transaction is still active.  I had the same issue and it took a lot of digging and frustration.  I looked for where I had opened files and not closed them in my code, make sure you are not doing auto-inc in clarion as this can cause a new connection to start and not close properly, and last but not least make sure if you are using MSSQL that you are using verifyViaSelect.  In the end, even after I made sure all my files were getting closed and the auto inc was on the server... the verifyViaSelect solved the last of the issue for me.  It seems that when it was using the stored procedure to validate the file structure clarion did not always close the connection after it was done with that task.

To troubleshoot these issues I added a few admin pages that displayed the "Active Transactions", and the "Cursors" with the last statement that generated both.  Following is the code for that. 

Active Transactions
-----------------------------------------------
Create View v_ActiveTransactions
as
SELECT
       [s_tst].[session_id],
       [s_es].[login_name] AS [Login Name],
       DB_NAME (s_tdt.database_id) AS [Database],
       [s_tdt].[database_transaction_begin_time] AS [Begin Time],
       [s_tdt].[database_transaction_log_bytes_used] AS [Log Bytes],
       [s_tdt].[database_transaction_log_bytes_reserved] AS [Log Rsvd],
       [s_est].text AS [Last T-SQL Text],
       [s_eqp].[query_plan] AS [Last Plan]
   FROM
       sys.dm_tran_database_transactions [s_tdt]
   JOIN
       sys.dm_tran_session_transactions [s_tst]
   ON
       [s_tst].[transaction_id] = [s_tdt].[transaction_id]
   JOIN
       sys.[dm_exec_sessions] [s_es]
   ON
       [s_es].[session_id] = [s_tst].[session_id]
   JOIN
       sys.dm_exec_connections [s_ec]
   ON
       [s_ec].[session_id] = [s_tst].[session_id]
   LEFT OUTER JOIN
    sys.dm_exec_requests [s_er]
   ON
       [s_er].[session_id] = [s_tst].[session_id]
   CROSS APPLY
       sys.dm_exec_sql_text ([s_ec].[most_recent_sql_handle]) AS [s_est]
   OUTER APPLY
       sys.dm_exec_query_plan ([s_er].[plan_handle]) AS [s_eqp]
   ORDER BY
       [Begin Time] ASC;

GO


Cursors
-------------------------
CREATE View [dbo].[dv_CursorSessions]
as
select  login_time,status,last_request_start_time,last_request_end_time,
ses.transaction_isolation_level, ses.lock_timeout,row_count, c.*, t.text
from sys.dm_exec_sessions ses
cross apply sys.dm_exec_cursors(ses.session_id) c
CROSS APPLY sys.dm_exec_sql_text (c.sql_handle) t
where ses.is_user_process=1  and ses.session_id <> @@SPID

GO

trent

  • Full Member
  • ***
  • Posts: 204
    • View Profile
    • Email
Re: SQL Sleeping Connections
« Reply #25 on: January 22, 2014, 01:59:45 PM »
Hi Kevin and Debra,

Thank you very much for your suggestions. What i'll try is this:

1) Add a 'CheckForLocks' routine to restart the WebServer when required on a timer in the WebServer procedure.
2) Add the VerifyViaSelect to the most heavily used tables using prop:sql and a select statement.

Hi Bruce,

Having 'CheckForLocks' routine as an option would be a great idea for anyone using an MSSQL database with their web app.

Regards,
Trent

kevin plummer

  • Hero Member
  • *****
  • Posts: 1195
    • View Profile
    • Production Accounting and Software Payroll
Re: SQL Sleeping Connections
« Reply #26 on: January 22, 2014, 04:26:53 PM »
Hi Debra,

how/where do you set the verifyviaselect?

My apps seem to run fine without it but always keen to explore better options.

Cheers,

Kev

debraballenger

  • Newbie
  • *
  • Posts: 49
    • View Profile
    • Email
Re: SQL Sleeping Connections
« Reply #27 on: January 22, 2014, 05:17:26 PM »
I have it on each table in my dictionary in the driver options /verifyviaselect.  I tried setting in at the connection in my web handler - process link embed but it does not seem to do the trick there...  but the dictionary does. 
This also made a significant performance increase when opening forms for one of my wider (many columns) tables.

trent

  • Full Member
  • ***
  • Posts: 204
    • View Profile
    • Email
Re: SQL Sleeping Connections
« Reply #28 on: February 03, 2014, 02:09:18 PM »
Hi Kevin and Debra,

I've tried quite a few things:

1) Check for locks: This didn't work for me. I had to keep adding more and more tables to check and they never returned any errors even though the WebServer was hanging.

2) Verify Via Select: I've added this into each table of one of the applications and haven't noticed any difference.

3) Killing all sleeping connections that are older than 1x hour: This has worked the best for me. I run this on a timer every 30 minutes. The only problem is that for some reason my WebServer is still GPF'ing at some stage that is rediculously hard to find. This is the code:

KillSleepingConnections ROUTINE
  clear(GSYSID:Record)
  GloSQLSysID{prop:sql} = 'DECLARE @v_spid INT '&|
    'DECLARE c_Users CURSOR '&|
    '  FAST_FORWARD FOR '&|
    '  SELECT SPID '&|
    '  FROM master..sysprocesses (NOLOCK) '&|
    '  WHERE spid>50 '&|
    '  AND status=''sleeping'' '&|
    '  AND DATEDIFF(mi,last_batch,GETDATE())>=60 '&|
    '  AND spid<>@@spid '&|
    'OPEN c_Users '&|
    'FETCH NEXT FROM c_Users INTO @v_spid '&|
    'WHILE (@@FETCH_STATUS=0) '&|
    'BEGIN '&|
    ' EXEC(''KILL ''+@v_spid) '&|
    ' FETCH NEXT FROM c_Users INTO @v_spid '&|
    'END '&|
    'CLOSE c_Users '&|
    'DEALLOCATE c_Users;'
  debug(GloSQLSysID{prop:sql})
  next(GloSQLSysID)
  if errorcode() = 90
    debug('Error on KillSleepingConnections, fileerror('& FILEERROR() &') fileerrorcode('& FILEERRORCODE() &')')
  ELSE
    debug('no error on KillSleepingConnections')
  end!If

4) Restart procedure to run every 12x hours: This I have just tried this morning. It's a basterdisation of the 'Check for Locks' code where the procedure doesn't check for locked tables, it will just restart the WebServer. There is a problem where the session data is not being saved into an XML file. The XML file is not even created. Here is the code:

  debug('RestartWebServer')
  p_web.RequestData.Webserver._wait()
  debug('Saving _SessionQueue to: '& clip(GBL:WorkDirectory) &'sessionqueue.xml')
  xml1.save(p_web.RequestData.Webserver._SessionQueue,clip(GBL:WorkDirectory) &'sessionqueue.xml','table','record')
  if xml1.Error
    debug('XML Error - '& xml1.Error &' '& xml1.ErrorStr)
  end!if
  debug('Saving _SessionDataQueue to: '& clip(GBL:WorkDirectory) &'sessiondataqueue.xml')
  xml1.save(p_web.RequestData.Webserver._SessionDataQueue,clip(GBL:WorkDirectory) &'sessiondataqueue.xml','table','record')
  if xml1.Error
    debug('XML Error - '& xml1.Error &' '& xml1.ErrorStr)
  end!if
  p_web.RequestData.Webserver._release()
  if exists(clip(GBL:WorkDirectory) &'sessionqueue.xml') and exists(clip(GBL:WorkDirectory) &'sessiondataqueue.xml')
    debug(Command('0') & command(''))
    Chain(Command('0') & command(''))
  ELSE
    debug('an xml file does not exist - restart cancelled')
  end!If

What I see in the debug is the first line 'Restart WebServer' then the Chain Command. I don't see any debug for the XML saving at all. Why wouldn't this debug be showing? (debug outputs to DebugView)

Regards,
Trent

debzidoodle

  • Jr. Member
  • **
  • Posts: 98
    • View Profile
    • Email
Re: SQL Sleeping Connections
« Reply #29 on: February 03, 2014, 04:08:11 PM »
Make sure you are not just checking for sleeping connections.  Look for Active Transactions.  Look at the code I (debraballenger) posted earlier.  Adding this as a view will let you easily see what code was executed to create that connection in most cases... sometimes it just tells you the cursor.  In the cases that it shows the cursor you are generally safe to kill the transaction without it killing your app. It is actually important for you to understand if you have sleeping connections or active transactions.  If they are active transactions killing the connection causes a roll back and you loose the data changes associated with that transaction.  So make sure you try to trace down any active transactions by looking at the statements that generated them.

I ended up writing a separate app that I can view stats of my main website, namely the historical response time, and memory utilized so I can spot spikes. And most importantly the active transactions using the view.  I filter the active transactions to show if the transaction has been open for more than 2 minutes.  The other things I have done is to turn on the logging that is found in the Multi Host app, and write a log of when my app restarts.  With the above history I can see what all was happening on the server at the time of the freeze.  One scenario that this helped me track down was in the log from the multi host app I could see that a request was sent to a particular field on a form, the server never sent back a response to the browser and the app hung.  So I knew exactly which field and form was causing my non responsive app and fixed it.  You can do the same thing with the active transactions based on the transaction start time, you will know what was happening on the server to create that transaction.
So in summary, create an admin site for yourself... the history it creates will save you a lot of frustration of tracking down impossible bugs.