Update Knowledge Articles in Bulk
Recently we had a big Knowledge Centric Support (KCS) initiative where we generated a ton of Articles for use in a Community. (We're still on Classic Knowledge.)
Problem is, some users forgot to tick the "Customer" box when creating the Articles. So those articles didn't appear to external users in the Community.
You can update these Articles in bulk, but you need to use a few obscure hooks in Apex. This method will create a draft of the Article, update the Knowledge Article Version, and then re-publish without creating a new Version.
The following code will update the 'IsVisibleInCsp' flag for a single Article specified in the 1st query, which will make it visible to Customers.
Here's a bulkified version. This is heavy on the CPU so I run it 10 at a time.
When going over these in bulk, you might run into a situation where there is an existing draft of an article that has been previously published; i.e., it's actively being updated. In this example, we search for those situations and don't create revisions of them.
This will go over all articles of type BA_Public_Knowledgebase__kav, 10 at a time. Update the code for your KnowledgeArticleVersion name, then run in Execute Anonymous. When you're done it will report that it updated 0 articles.
Problem is, some users forgot to tick the "Customer" box when creating the Articles. So those articles didn't appear to external users in the Community.
You can update these Articles in bulk, but you need to use a few obscure hooks in Apex. This method will create a draft of the Article, update the Knowledge Article Version, and then re-publish without creating a new Version.
The following code will update the 'IsVisibleInCsp' flag for a single Article specified in the 1st query, which will make it visible to Customers.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// In this example my Knowledge Article Type is "BA Public Knowledgebase" (BA_Public_Knowledgebase__kav) | |
// Query for the online version; filter by KA version is optional. | |
List<BA_Public_Knowledgebase__kav> baKA = [select id, IsVisibleInCsp, publishstatus, knowledgearticleid from BA_Public_Knowledgebase__kav where PublishStatus = 'Online' and ID = 'kaA2M000003PCxs']; | |
Id kaId = baKA[0].KnowledgeArticleId; | |
// Create a revision without unpublishing. | |
KbManagement.PublishingService.editOnlineArticle(baKA[0].KnowledgeArticleId, false); | |
// Requery to get the new version ID. | |
baKA = [SELECT Id, KnowledgeArticleId, PublishStatus, IsVisibleInCsp FROM BA_Public_Knowledgebase__kav WHERE KnowledgeArticleId = : kaId and PublishStatus = 'Draft' limit 1]; | |
// Set to be visible to Customers. There is also IsVisibleInPrm (Partner Community), IsVisibleInApp (internal app), and IsVisibleInPkb (Public KnowledgeBase) | |
BA_Public_Knowledgebase__kav newKAV = new BA_Public_Knowledgebase__kav(id = baKA[0].Id, IsVisibleInCsp = true); | |
update newKav; | |
// Republish without a new version | |
KbManagement.PublishingService.publishArticle(kaId, false); |
Here's a bulkified version. This is heavy on the CPU so I run it 10 at a time.
When going over these in bulk, you might run into a situation where there is an existing draft of an article that has been previously published; i.e., it's actively being updated. In this example, we search for those situations and don't create revisions of them.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// Query for the online versions | |
// Since this is heavy, process 10 at a time. | |
Map<Id,BA_Public_Knowledgebase__kav> kavsToProcess = new Map<Id,BA_Public_Knowledgebase__kav>([select id, Title, IsVisibleInCsp, publishstatus, knowledgearticleid from BA_Public_Knowledgebase__kav where IsLatestVersion = true and publishstatus = 'Online' and IsVisibleInCsp = false limit 10]); | |
// Run over the KA Versions and collect the KnowledgeArticle ID's. | |
Set<Id> kaIds = new Set<Id>(); | |
for (BA_Public_Knowledgebase__kav thisKAV : kavsToProcess.values()){ | |
kaIds.add(thisKAV.KnowledgeArticleId); | |
} | |
// Query for EXISTING draft versions of our published articles. We'll exclude them from our updates. | |
List<BA_Public_Knowledgebase__kav> draftKAVs = [select id, Title, IsVisibleInCsp, publishstatus, knowledgearticleid from BA_Public_Knowledgebase__kav where PublishStatus = 'Draft' and KnowledgeArticleId in : kaIds]; | |
List<BA_Public_Knowledgebase__kav> toUpdate = new List<BA_Public_Knowledgebase__kav>(); | |
// Go over the KnowledgeArticleVersions from the 2nd query and create a list (Set) of KnowledgeArticle ID's that | |
// have an existing draft. | |
Set<Id> kasWithDrafts = new Set<Id>(); | |
for (BA_Public_Knowledgebase__kav draftKAV : draftKAVs){ | |
kasWithDrafts.add(draftKav.KnowledgeArticleId); | |
System.debug('Eliminating Article From Processing, Already Has Draft: ' + draftKav.Title + ' ' + draftKav.id); | |
// If this version doesn't have our flag, add it to the list to update. | |
if (draftKav.IsVisibleInCsp == false){ | |
toUpdate.add(new BA_Public_Knowledgebase__kav(Id = draftKAV.Id, IsVisibleInCsp = true)); | |
} | |
} | |
Integer quantityPreExistingDrafts = toUpdate.size(); | |
System.debug('Will update ' + quantityPreExistingDrafts + ' pre-existing drafts.'); | |
// Now, go back over the original list of published versions and create an un-versioned revision of any Article that does not | |
// have a pre-existing draft. | |
for (BA_Public_Knowledgebase__kav thisKAV : kavsToProcess.values()){ | |
if (!kasWithDrafts.contains(thisKav.KnowledgeArticleId)) { | |
System.debug('Updating ' + thisKav.Title); | |
KbManagement.PublishingService.editOnlineArticle(thisKAV.KnowledgeArticleId, false); | |
} | |
} | |
// Requery to get all the Knowledge Article VERSION Id's created above from editOnlineArticle(). | |
// Note that this EXCLUDES the KnowledgeArticles that already have a draft. | |
kavsToProcess = new Map<Id,BA_Public_Knowledgebase__kav>([SELECT Id, KnowledgeArticleId, PublishStatus, IsVisibleInCsp FROM BA_Public_Knowledgebase__kav WHERE KnowledgeArticleId IN : kaIds and PublishStatus = 'Draft' and (NOT(Id IN : kasWithDrafts))]); | |
// Go over all our newly created versions, and update them with the attribute we're going for (IsVisibleInCsp) | |
for (BA_Public_Knowledgebase__kav thisKAV : kavsToProcess.values()) { | |
toUpdate.add(new BA_Public_Knowledgebase__kav (Id = thisKAV.Id, IsVisibleInCsp = true)); | |
} | |
update toUpdate; | |
// Now publish the new versions that we have updated. | |
for (BA_Public_Knowledgebase__kav thisKAV : kavsToProcess.values()){ | |
KbManagement.PublishingService.publishArticle(thisKAV.KnowledgeArticleId, false); | |
} | |
System.debug('Total Updated: ' + toUpdate.size()); | |
System.debug('... ' + quantityPreExistingDrafts + ' pre-existing drafts were updated'); | |
System.debug('... ' + (toUpdate.size()-quantityPreExistingDrafts) + ' were republished'); | |
System.debug('Success'); |
This is awesome. It helped me do the same for one of our requirements. I am facing one challenge though and unable to figure it out. It worked seamlessly for one article type but not working for any other article types. I am a system admin user and have all access to publish articles. Not sure what is the reason. Keep getting the below error:
ReplyDeleteSystem.HandledException: You can't perform this action. Be sure the action is valid for the current state of the article, and that you have permission to perform it.
hi, I have a question for this method KbManagement.PublishingService.editOnlineArticle, I tried to replace the current online article version with a new one, but it seems that the editOnlineArticle with param unpublish set to TRUE do not works as expected creating a new version record (__KAV) with status Draft, on the contrary I got an unexpected behaviour, the method only update the current online version record to a new version number and changed the publish status field to Draft. Should this be the right logic? I went through the standard interface to create a new article version and from this action I got a new __KAV record with version 0 and publish status as Draft, having at the end two records (__KAV) for the same article, on for Draft and another for Online. I am just wandering if this is the default logic when creating versions from apex. thxs!
ReplyDelete