In SharePoint timer jobs there is a simple way to implement app setting like key value pairs which can be used to store configuration values. Since Timer Jobs run under SharePoint central admin context so keeping configuration values in web.config is not feasible and that too if a job is catering to multiple websites then keeping web specific configuration is also a challenge.
As a simple yet effective solution is to keep such configuration in the form of a custom list at web or site collection level, whichever is more feasible and logical to your need. Here is how I have done the same
For my requirement I need to have couple of static values and some environment specific configurations to be stored for use in a Timer job. As a solution to this I decided to keep these configurations in a custom defined list. To do that
Create a custom list and name it Config or Configuration in the web site and add one field "Value" Set the type of this field to multiple lines of text( set rich text to false.
This list will store all the configurable values e.g. environment specific server paths, list names, notification email formats, email addresses or whatever you want to store as a configuration.
After defining this list now next step is to populate the configuration values and use them in the timer job.
CustomTimerJob.cs
using System;
using System.Data;
using System.Text;
using System.Collections.Generic;
using Microsoft.SharePoint;
using Microsoft.SharePoint.Administration;
using Microsoft.Office.Server.Diagnostics;
public class MyTimerJob : SPJobDefinition
{
private static string SERVER_NAME = "xxx-xxx-xxxx";
private static string LIST_NAME = "Test";
private static string WEB_NAME = "Webname";
public override void Execute(Guid targetInstanceId)
{
try
{
SPSecurity.RunWithElevatedPrivileges(delegate()
{
SPWebApplication webApp = this.Parent as SPWebApplication;
//root site collection
string siteURL = webApp.Sites[0].Url;
using (SPSite site = new SPSite(siteURL))
{
using (SPWeb web = site.AllWebs[WEB_NAME])
{
//Read timer job configuration
//from config list from web
ReadConfiguration(web);
SPList list = web.Lists[LIST_NAME];
SPQuery q = new SPQuery();
//Get all items to process
q.Query = "<Where><And><Eq><FieldRef Name='WFStatus' /><Value Type='Text'>3</Value></Eq><Eq><FieldRef Name='IsProcessed' /><Value Type='Boolean'>0</Value></Eq></And></Where><OrderBy><FieldRef Name='ID' /></OrderBy>";
//Record collection
SPListItemCollection itemCollection = list.GetItems(q);
if (itemCollection.Count > 0)
{
//Write your logic here for items
}
}
}
});
}
catch (Exception ex)
{
PortalLog.LogString("Exception Occurred in: {0} || {1} || {2}", " Timer Job: Execute Method", ex.Message, ex.StackTrace);
}
}
/// <summary>
/// Read configuration from config list from web.
/// </summary>
/// <param name="web">Web object</param>
private static void ReadConfiguration(SPWeb web)
{
//Overwrite default values from configuration
if (web != null)
{
SPList configList = web.Lists.TryGetList("Config");
if (configList != null)
{
foreach (SPListItem item in configList.Items)
{
SERVER_NAME = GetConfigValue(item, "SERVER_NAME")!= string.Empty ? GetConfigValue(item, "SERVER_NAME") : SERVER_NAME;
LIST_NAME = GetConfigValue(item, "LIST_NAME")!= string.Empty ? GetConfigValue(item, "LIST_NAME") : LIST_NAME;
}
}
}
}
/// <summary>
/// Get config list value based on key.
/// </summary>
/// <param name="key">key name</param>
/// <returns></returns>
private static string GetConfigValue(SPListItem item, string key)
{
string value = string.Empty;
if (string.Equals(item["Title"].ToString(), key) && item["Value"] != null)
{
value = item["Value"].ToString();
}
return value;
}
}
The above code will read the configuration values from the config list from the web. To make this config list non editable be normal users stop the inheritance level for the list and add only site collection admin or other admin user contribute access to the list and also add the farm account user under which the timer job will run to the list contribute permissions as well. When the job tried to access this list the user under which the timer job is running should have access to the list.
Having this type of configuration has key benefits of
1) Managing the config values with ease and no physical file access required like the web.config file.
2) Change in config values does not impact the AppPool or resets web application.
3) Provides more control, for example if you need to write a job to send email notifications then the email format can be kept as a config value and changed anytime without impacting code.
4) Make the job as configurable as possible so that minimal code changes are required.
Happy Coding!
As a simple yet effective solution is to keep such configuration in the form of a custom list at web or site collection level, whichever is more feasible and logical to your need. Here is how I have done the same
For my requirement I need to have couple of static values and some environment specific configurations to be stored for use in a Timer job. As a solution to this I decided to keep these configurations in a custom defined list. To do that
Create a custom list and name it Config or Configuration in the web site and add one field "Value" Set the type of this field to multiple lines of text( set rich text to false.
This list will store all the configurable values e.g. environment specific server paths, list names, notification email formats, email addresses or whatever you want to store as a configuration.
After defining this list now next step is to populate the configuration values and use them in the timer job.
CustomTimerJob.cs
using System;
using System.Data;
using System.Text;
using System.Collections.Generic;
using Microsoft.SharePoint;
using Microsoft.SharePoint.Administration;
using Microsoft.Office.Server.Diagnostics;
public class MyTimerJob : SPJobDefinition
{
private static string SERVER_NAME = "xxx-xxx-xxxx";
private static string LIST_NAME = "Test";
private static string WEB_NAME = "Webname";
public override void Execute(Guid targetInstanceId)
{
try
{
SPSecurity.RunWithElevatedPrivileges(delegate()
{
SPWebApplication webApp = this.Parent as SPWebApplication;
//root site collection
string siteURL = webApp.Sites[0].Url;
using (SPSite site = new SPSite(siteURL))
{
using (SPWeb web = site.AllWebs[WEB_NAME])
{
//Read timer job configuration
//from config list from web
ReadConfiguration(web);
SPList list = web.Lists[LIST_NAME];
SPQuery q = new SPQuery();
//Get all items to process
q.Query = "<Where><And><Eq><FieldRef Name='WFStatus' /><Value Type='Text'>3</Value></Eq><Eq><FieldRef Name='IsProcessed' /><Value Type='Boolean'>0</Value></Eq></And></Where><OrderBy><FieldRef Name='ID' /></OrderBy>";
//Record collection
SPListItemCollection itemCollection = list.GetItems(q);
if (itemCollection.Count > 0)
{
//Write your logic here for items
}
}
}
});
}
catch (Exception ex)
{
PortalLog.LogString("Exception Occurred in: {0} || {1} || {2}", " Timer Job: Execute Method", ex.Message, ex.StackTrace);
}
}
/// <summary>
/// Read configuration from config list from web.
/// </summary>
/// <param name="web">Web object</param>
private static void ReadConfiguration(SPWeb web)
{
//Overwrite default values from configuration
if (web != null)
{
SPList configList = web.Lists.TryGetList("Config");
if (configList != null)
{
foreach (SPListItem item in configList.Items)
{
SERVER_NAME = GetConfigValue(item, "SERVER_NAME")!= string.Empty ? GetConfigValue(item, "SERVER_NAME") : SERVER_NAME;
LIST_NAME = GetConfigValue(item, "LIST_NAME")!= string.Empty ? GetConfigValue(item, "LIST_NAME") : LIST_NAME;
}
}
}
}
/// <summary>
/// Get config list value based on key.
/// </summary>
/// <param name="key">key name</param>
/// <returns></returns>
private static string GetConfigValue(SPListItem item, string key)
{
string value = string.Empty;
if (string.Equals(item["Title"].ToString(), key) && item["Value"] != null)
{
value = item["Value"].ToString();
}
return value;
}
}
The above code will read the configuration values from the config list from the web. To make this config list non editable be normal users stop the inheritance level for the list and add only site collection admin or other admin user contribute access to the list and also add the farm account user under which the timer job will run to the list contribute permissions as well. When the job tried to access this list the user under which the timer job is running should have access to the list.
Having this type of configuration has key benefits of
1) Managing the config values with ease and no physical file access required like the web.config file.
2) Change in config values does not impact the AppPool or resets web application.
3) Provides more control, for example if you need to write a job to send email notifications then the email format can be kept as a config value and changed anytime without impacting code.
4) Make the job as configurable as possible so that minimal code changes are required.
Happy Coding!