1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Semantic Kernelで社内向けAIアシスタントを作った

Posted at

こんにちは。

テックリードのTerukiです。

世間ではAIが大変流行っていますが、社内でも便利に使えるようなAIアシスタントが欲しくなっているところです。

Azure OpenAIは契約済みですが、これだけあってもしょうがないのでずっと前から気になっていたSemantic Kernelというフレームワークを使って社内向けAIアシスタントを作ってみます。

Semantic Kernel

Semantic Kernel is a lightweight, open-source development kit that lets you easily build AI agents and integrate the latest AI models into your C#, Python, or Java codebase. It serves as an efficient middleware that enables rapid delivery of enterprise-grade solutions.

軽くてオープンソースのAIエージェントやモデルを簡単に組み込めるエンタープライズグレードのソリューションである、くらいでしょうか。

Microsoftの@okazukiさんの記事は結構読み込んでいるつもりですが、実際に触るのは今回が初めてなのであんまりよく分かっていません。

Slackに繋いでみる

単にSlackでGPTとやり取りできるできるだけでは劣化版ChatGPTなので社内特有の何かができるようなものにしたいところです。

ただ、それをやる前にまずはSlackと繋いでみます。

いつ追加されたのか不明ですが、SlackアプリにAIエージェント用の機能が追加されています。

image.png

これを有効にするとSlack上にいつでもAIエージェントを呼び出せるボタンが追加できます。

イメージ
image.png

これはこれで便利ですね。

これだけでは物足りないので、Notionにある情報を取りに行かせてみます。
NotionにはFAQが保存されているとします。

var kernelBuilder = Kernel.CreateBuilder()
    .AddAzureOpenAIChatCompletion(Configuration["Azure:OpenAI:DeploymentName"]!,
        Configuration["Azure:OpenAI:Endpoint"]!, Configuration["Azure:OpenAI:ApiKey"]!);
    
var kernel = kernelBuilder.Build();

// INotionServiceを解決するためにアプリのIServiceProviderを指定しておく
kernel.Plugins.AddFromType<NotionPlugin>(null, ServiceProvider);

var agent = new ChatCompletionAgent {
    Name = "Agent",
    Instructions = """
        質問への応答や回答を考えて返答してください。
        FAQNotionに保存されています。
        """,
    InstructionsRole = AuthorRole.Assistant,
    Kernel = kernel,
    Arguments = new(new AzureOpenAIPromptExecutionSettings {
        FunctionChoiceBehavior = FunctionChoiceBehavior.Auto(),
    }),
};

var thread = new ChatHistoryAgentThread();

// Agent を呼び出し
await foreach (var agentResult in agent.InvokeAsync(new ChatMessageContent(AuthorRole.User, text), thread)) {
    // ~~
    // agentResult.Message.ContentをSlackに投稿
}

NotionPluginのイメージはこんな感じです。

public class NotionPlugin(INotionService NotionService) {

    [KernelFunction]
    [Description("FAQを取得する。")]
    public async Task<string[]> QueryFaqAsync() {

        var response = await NotionService.QueryDatabaseAsync("NotionデータベースのID").ConfigureAwait(false);

        return response.Where(w => w["有効"] == "true")
            .Select(s => $"""
            タイトル:{s["タイトル"]}
            分類:{s["分類"]}
            内容:
            {s["内容"]}
            """).ToArray();
    }
}

NotionServiceが実際にNotionのAPIを叩いているところです。
有効のチェックボックスが入っているもののみ絞るようにしてみました。
(本来はNotion APIのパラメータとして渡すべきではあります)

Notionデータベースにはこんな感じでデータを入れてみました。

image.png

パースが大変なのでページ本文には何も書かずにプロパティに内容を書いてしまうのが良いです。

結果がこちら。
image.png

ちゃんと内容を汲み取って返信してきていますね!

ただ、この実装ではSlackのスレッドの文脈を全く覚えていないのでさすがにそこは覚えて欲しいところです。

ということでChatHistoryをIDistributedCacheに入れて処理を終えた後に保存するようにします。

SlackはスレッドIDというタイムスタンプっぽいIDを持っているのでそれを使うと良さそうです。
(厳密にやるならチャンネルIDとセットにしたほうが良さそうです。)

var kernelBuilder = Kernel.CreateBuilder()
    .AddAzureOpenAIChatCompletion(Configuration["Azure:OpenAI:DeploymentName"]!,
        Configuration["Azure:OpenAI:Endpoint"]!, Configuration["Azure:OpenAI:ApiKey"]!);
    
var kernel = kernelBuilder.Build();

kernel.Plugins.AddFromType<NotionPlugin>(null, ServiceProvider);

var agent = new ChatCompletionAgent {
    Name = "Agent",
    Instructions = $"""
        質問への応答や回答を考えて返答してください。あまり堅すぎないトーンにすること。
        返信を考える時はユーザーのステータス、メッセージ履歴、FAQを必ず参照し、
        {(user is not null ? "\nユーザーID: " + user.ScreenId : "")}
        """.Trim(),
    InstructionsRole = AuthorRole.Assistant,
    Kernel = kernel,
    Arguments = new(new AzureOpenAIPromptExecutionSettings {
        FunctionChoiceBehavior = FunctionChoiceBehavior.Auto(),
    }),
};

var cacheHistory = await Cache.GetStringAsync(threadId).ConfigureAwait(false);
var chatHistory = cacheHistory is not null ? JsonSerializer.Deserialize<ChatHistory>(cacheHistory) : null;

// Thread を作成
var thread = chatHistory is not null ? new ChatHistoryAgentThread(chatHistory) : new ChatHistoryAgentThread();

// Agent を呼び出し
await foreach (var agentResult in agent.InvokeAsync(new ChatMessageContent(AuthorRole.User, text), thread)) {
    // ~~
    // agentResult.Message.ContentをSlackに投稿
}

// ChatHistoryをキャッシュに保存
var json = JsonSerializer.Serialize(thread.ChatHistory);
await Cache.SetStringAsync(threadId, json, new DistributedCacheEntryOptions {
    AbsoluteExpirationRelativeToNow = TimeSpan.FromMinutes(30)
}).ConfigureAwait(false);

これで試してみます。

image.png

1つ目のメッセージではネットの情報から回答をしていますが、2つ目のメッセージはFAQを参照していますね!

私の2つ目のメッセージにOh my teethという文字列が含まれていないのでしっかり文脈を認識できていることが分かります。


まだまだ使い始めたばかりですが、しっかり作り込んだらめちゃくちゃ便利なAIアシスタントが作れそうです。
ちなみに今回はGPT4.1を使用しています。

インターフェースがSlackなのでやろうと思えば押せるボタンを追加して追加処理をさせるみたいなこともできそうです。

今後もどんどんアップデートしていこうと思っています。

Oh my teethについて

Oh my teethでは未来の歯科体験を創るために日々活動しています。

Techチームではより良いユーザー体験を提供するべく、Webフロントエンドからバックエンド、スマホアプリに機械学習モデルなど、さまざまなプロダクトを開発しています。

一緒に未来の歯科体験を創りませんか?興味がある方は是非こちらを確認してください。

カジュアル面談も可能なので気軽に応募してみてください!

1
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?