Provisioning a QnA Maker knowledge base programmatically
This blog post is about the new Global Availability version of the QnA Maker service. You can read the announcement at the Bot Framework blog.
During the past week, while working with one of my customers, I faced a challenge that seemed easy at first glance but proved to be a much harder nut to crack when I started to look into the details. I spent a good portion of a day figuring this stuff out, so I hope I can spare you that amount of time by writing this blog post.
The challenge was the following: I needed to provision a QnA Maker service instance, create a knowledge base on top of it, and then publish it, doing all this in an automated way. Sounds easy, right? Actually, it is: we have ARM templates, nice APIs, all those good things. The big challenge was that I had to make sure that the language analyzers of the underlying Azure Search were set correctly to the language of the knowledge base. This is where things got a little bit complicated.
But let’s not get ahead of ourselves and start at the beginning!
The business reason #
The QnA Maker is a nice service built with non-technical users in mind. On the QnA Maker portal, these users can go ahead and create their own knowledge bases and train their own services without understanding the details of how the underlying intelligence itself works.
As part of a bigger solution using other Azure services, that’s exactly what we wanted to use the QnA Maker for. But we also wanted to make sure to present the users with a knowledge base seeded with some default content when they first open the portal, thus minimizing the chance of these users accidentally creating a knowledge base with incorrect settings.
Why all this hassle? What could go wrong? #
To see clearly the core of the problem, you first need to read this page of the docs. These are the key pieces of information from there:
QnA Maker supports knowledge base content in many languages. However, each QnA Maker service should be reserved for a single language. The first knowledge base created targeting a particular QnA Maker service sets the language of that service.
…
The language is automatically recognized from the content of the data sources being extracted.
…
This language cannot be changed once the resource is created.
What this ultimately means is that you either get the language setting right at the very beginning, or you are screwed.
The solution #
These are the steps you need to follow if you want to be 100% sure that the language of the underlying service of your knowledge base will be set correctly as part of a fully programmatic deployment.
1. Create the QnA Maker service with an ARM template #
Azure Resource Manager templates are pretty cool. You can describe the services and the relationships between those services that you want to deploy to Azure in a totally readable (and versionable!) JSON file and with a simple command deploy all of them in one step. You can read more about Azure Resource Manager and ARM templates here.
The easiest way to get such an ARM template for a QnA Maker Service is to first deploy one manually to Azure and export the template from the resource group that encapsulates it.
2. Create a knowledge base with dummy data via the API #
This was the bit which took most of my time to figure out. As the documentation states (see quote above), when you first create a knowledge base for a QnA Maker service, it will try to figure out the language of the knowledge base and set the analyzer’s language in the underlying Azure Search accordingly. What I discovered while playing around with the API is that this language recognition works reliably only for “big enough” data sets of question and answer pairs. It works for a single pair in the rarest of cases, mostly just falling back to English, which is obviously not acceptable at all.
So let’s say you create a knowledge base with a single QnA pair using the appropriate API, with the HTTP request’s body looking like this:
{
"name": "Italian QnA Maker KB",
"qnaList": [
{
"id": 0,
"answer": "ciao",
"source": "Editorial",
"questions": ["ciao", "buongiorno", "buona sera"]
}
]
}
No matter how Italian it looks to you, QnA Maker will happily set the underlying Azure Search’s analyzers to English.
The way we can fix this is by feeding QnA Maker a “big enough” set of data initially. Then it will pick up its language correctly. I really don’t have a concrete number of question and answer pairs with which it will work correctly. My workaround for this was (somewhat ironically) to use the FAQ page of QnA Maker itself, always changing its language accordingly. So if we want to create a knowledge base and want to make super sure that the underlying service’s language will be set to Italian, we call the API with the following body:
{
"name": "Italian QnA Maker KB",
"urls": [
"https://azure.microsoft.com/it-it/services/cognitive-services/qna-maker/faq/"
]
}
The only thing we change between languages is the culture code in the URL. In the above example, it’s it-it
. The same part for Hungarian would be hu-hu
and so on.
3. Replace the knowledge base with the correct default data via the API #
If we created the knowledge base with the above data, the analyzers will be set correctly, but the first users to start using it will find a lot of question and answer pairs that they have nothing to do with. (Except if they want to create a chatbot for the QnA Maker itself, but that’s a very-very rare case among my customer’s users.)
To fix this, we need to replace the previously created knowledge base’s content (calling the appropriate REST API) with something more consumable, like this:
{
"qnaList": [
{
"id": 0,
"answer": "ciao",
"source": "Editorial",
"questions": ["ciao", "buongiorno", "buona sera"]
}
]
}
4. (Optional) Publish the knowledge base via the API #
This step is optional and depends on which state you plan to hand over the freshly created knowledge base to your users. In our case - as I mentioned before - QnA Maker was just part of a bigger solution, so we wanted to make sure that every part of it was available before giving the whole thing to the users to play around with.
Luckily, doing this is very easy. All you need to do is just POST the publish knowledge base endpoint.
Conclusion #
To wrap all this in a nice and concise way, you can easily write a small script that deploys the ARM template via the Azure Resource Manager API and then calls the above three QnA Maker APIs in order. A great starting point for such a utility script could be the exported ARM template package, since that already contains not just the ARM template itself, but also a couple of lines of code in different languages/command line tools (currently Azure CLI, PowerShell, C#, Ruby). Working on top of that, it’s easy to implement calling the QnA Maker APIs with the correct seed values.
After all is said and done, you can provide your users with a QnA Maker environment that you can be sure is set up correctly to support their choice of language.