How to Identify whether SQL job is successfully executed or not in C#

后端 未结 3 1753
滥情空心
滥情空心 2020-12-06 14:30

I have an C# method to execute a SQL job. It executes the SQL job successfully. And the code works perfect.

And I\'m using standard SQL stored procedure msdb.d

相关标签:
3条回答
  • 2020-12-06 14:41

    You need to run stored proceedure msdb.dbo.sp_help_job

         private int CheckAgentJob(string connectionString, string jobName) {
            SqlConnection dbConnection = new SqlConnection(connectionString);
            SqlCommand command = new SqlCommand();
            command.CommandType = System.Data.CommandType.StoredProcedure;
            command.CommandText = "msdb.dbo.sp_help_job";
            command.Parameters.AddWithValue("@job_name", jobName);
            command.Connection = dbConnection;
            using (dbConnection)
            {
                dbConnection.Open();      
                using (command){
                    SqlDataReader reader = command.ExecuteReader();
                    reader.Read();
                    int status = reader.GetInt32(21); // Row 19 = Date Row 20 = Time 21 = Last_run_outcome
                    reader.Close();
                    return status;
                }
            }
         }
    
    enum JobState { Failed = 0, Succeeded = 1, Retry = 2, Cancelled = 3, Unknown = 5};
    

    Keep polling on Unknown, until you get an answer. Lets hope it is succeeded :-)

    0 讨论(0)
  • 2020-12-06 14:45

    After running msdb.dbo.sp_start_job the return code is mapped to an output parameter. You have the opportunity to control the parameter's name prior to execution:

    public int StartMyJob( string connectionString )
    {
     using (var sqlConnection = new SqlConnection( connectionString ) )
     {
       sqlConnection.Open( );
       using (var execJob = sqlConnection.CreateCommand( ) )
       {
          execJob.CommandType = CommandType.StoredProcedure;
          execJob.CommandText = "msdb.dbo.sp_start_job";
          execJob.Parameters.AddWithValue("@job_name", "myjobname");
          execJob.Parameters.Add( "@results", SqlDbType.Int ).Direction = ParameterDirection.ReturnValue;      
          execJob.ExecuteNonQuery();
          return ( int ) sqlCommand.Parameters["results"].Value;
        }
      }
    }
    

    You need to know the datatype of the return code to do this - and for sp_start_job, it's SqlDbType.Int.

    However, this is only the results of starting the job, which is worth knowing, but isn't the results of running your job. To get the results running of your job, you can periodically execute:

    msdb.dbo.sp_help_job @jobName
    

    One of the columns returned by the procedure is last_run_outcome and probably contains what you're really interested in. It will be 5 (unknown) while it's still running.

    A job is usually the a number of steps - where each step may or may not be executed according to the outcome of previous steps. Another procedure called sp_help_jobhistory supports a lot of filters to specify which specific invocation(s) and/or steps of the job you're interested in.

    SQL likes to think about jobs as scheduled work - but there's nothing to keep you from just starting a job ad-hoc - although it doesn't really provide you with much support to correlate your ad-hoc job with an instance is the job history. Dates are about as good as it gets (unless somebody knows a trick I don't know.)

    I've seen where the job is created ad-hoc job just prior to running it, so the current ad-hoc execution is the only execution returned. But you end up with a lot of duplicate or near-duplicate jobs laying around that are never going to be executed again. Something you'll have to plan on cleaning up afterwards, if you go that route.

    A note on your use of the _sqlConnection variable. You don't want to do that. Your code disposes of it, but it was apparently created elsewhere before this method gets called. That's bad juju. You're better off just creating the connection and disposing of it the same method. Rely on SQL connection pooling to make the connection fast - which is probably already turned on.

    Also - in the code you posted - it looks like you started with execJob but switched to sqlCommand - and kinda messed up the edit. I assumed you meant execJob all the way through - and that's reflected in the example.

    0 讨论(0)
  • 2020-12-06 14:58

    From MSDN about SqlCommand.ExecuteNonQuery Method:

    For UPDATE, INSERT, and DELETE statements, the return value is the number of rows affected by the command. When a trigger exists on a table being inserted or updated, the return value includes the number of rows affected by both the insert or update operation and the number of rows affected by the trigger or triggers. For all other types of statements, the return value is -1. If a rollback occurs, the return value is also -1.

    In this line:

    result = sqlCommand.ExecuteNonQuery();
    

    You want to return the number of rows affected by the command and save it to an int variable but since the type of statement is select so it returns -1. If you test it with INSERT or DELETE or UPDATE statements you will get the correct result.

    By the way if you want to get the number of rows affected by the SELECT command and save it to an int variable you can try something like this:

    select count(*) from jobs where myjobname = @myjobname
    

    And then use ExecuteScalar to get the correct result:

    result = (int)execJob.ExecuteScalar();
    
    0 讨论(0)
自定义标题
段落格式
字体
字号
代码语言
提交回复
热议问题