Programmatically compile Audiences in SharePoint 2010

According to the documentation for the SharePoint 2010 SDK, you cannot trigger Audience compilation programmatically. Well—that’s not true. It isn’t documented, but it is possible. The painful thing about it is that you must know the ApplicationId of the UserProfileService application. This is where reflection comes in extremely handy!

Below, you will find two incredibly similar snippets of code: one C# example and one PowerShell example. They both focus on reflecting to grab the ApplicationId and then using the RunAudienceJob method to queue the compilation. I found many examples online that explain the method call, and many that showed how to use reflection to pull out the ApplicationId, but none of the articles I found showed the whole picture from start to finish. Here you go.

C# server-side code:

	// get the assembly which hosts the UserProfile class
	Assembly userProfilesAssembly = typeof(UserProfile).Assembly;
	// get the type of the UserProfileApplicationProxy
	Type userProfileApplicationProxyType = userProfilesAssembly.GetType("Microsoft.Office.Server.Administration.UserProfileApplicationProxy");
	// get the proxy object
	object proxy = SPServiceContext.Current.GetDefaultProxy(userProfileApplicationProxyType);
	// get the UserProfileApplication property which holds the actual application
	object profile = proxy.GetType().GetProperty("UserProfileApplication", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(proxy, null);
	// get the Id of the application
	Guid applicationId = ((Microsoft.SharePoint.Administration.SPPersistedObject)profile).Id;

	string[] args = new string[4];
	// application ID of UserProfileApplication
	args[0] = applicationId.ToString();
	// 1 = start, 0 = stop
	args[1] = "1";
	// 1 = full, 0 = incremental
	args[2] = "0";
	// audience name
	args[3] = "Audience Name";

	// compile the audience
	int result = Microsoft.Office.Server.Audience.AudienceJob.RunAudienceJob(args);
	AudienceJobReturnCode returnCode = (AudienceJobReturnCode)Enum.Parse(typeof(AudienceJobReturnCode), result.ToString());

Note: You must run the PowerShell script as a Farm Administrator user, or you will be unsuccessful!

PowerShell command-line code:

if((Get-PSSnapin "Microsoft.SharePoint.PowerShell" -ErrorAction SilentlyContinue) -eq $null) {
	Add-PSSnapin "Microsoft.SharePoint.PowerShell"

$site = Get-SPSite ""
$audience = "Audience Name"
$context = Get-SPServiceContext $site  
$proxy = $context.GetDefaultProxy([Microsoft.Office.Server.Audience.AudienceJob].Assembly.GetType("Microsoft.Office.Server.Administration.UserProfileApplicationProxy")) 
$applicationId = $proxy.GetType().GetProperty("UserProfileApplication", [System.Reflection.BindingFlags]"NonPublic, Instance").GetValue($proxy, $null).Id.Guid 
[Array]$args = $applicationId, "1", "0", $audience
[int]$runjob = [Microsoft.Office.Server.Audience.AudienceJob]::RunAudienceJob($args)

(For an explanation of $args in that PowerShell snippet, see the C# example.)