forked from microsoft/semantic-kernel
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathBing_TextSearch.cs
More file actions
215 lines (191 loc) · 9.05 KB
/
Bing_TextSearch.cs
File metadata and controls
215 lines (191 loc) · 9.05 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
// Copyright (c) Microsoft. All rights reserved.
using System.Text.Json;
using Microsoft.SemanticKernel.Data;
using Microsoft.SemanticKernel.Plugins.Web.Bing;
namespace Search;
/// <summary>
/// This example shows how to create and use a <see cref="BingTextSearch"/>.
/// </summary>
public class Bing_TextSearch(ITestOutputHelper output) : BaseTest(output)
{
/// <summary>
/// Show how to create a <see cref="BingTextSearch"/> and use it to perform a text search.
/// </summary>
[Fact]
public async Task UsingBingTextSearchAsync()
{
// Create a logging handler to output HTTP requests and responses
LoggingHandler handler = new(new HttpClientHandler(), this.Output);
using HttpClient httpClient = new(handler);
// Create an ITextSearch instance using Bing search
var textSearch = new BingTextSearch(apiKey: TestConfiguration.Bing.ApiKey, options: new() { HttpClient = httpClient });
var query = "What is the Semantic Kernel?";
// Search and return results as a string items
KernelSearchResults<string> stringResults = await textSearch.SearchAsync(query, new() { Top = 4, Skip = 0 });
Console.WriteLine("--- String Results ---\n");
await foreach (string result in stringResults.Results)
{
Console.WriteLine(result);
WriteHorizontalRule();
}
// Search and return results as TextSearchResult items
KernelSearchResults<TextSearchResult> textResults = await textSearch.GetTextSearchResultsAsync(query, new() { Top = 4, Skip = 4 });
Console.WriteLine("\n--- Text Search Results ---\n");
await foreach (TextSearchResult result in textResults.Results)
{
Console.WriteLine($"Name: {result.Name}");
Console.WriteLine($"Value: {result.Value}");
Console.WriteLine($"Link: {result.Link}");
WriteHorizontalRule();
}
// Search and return s results as BingWebPage items
KernelSearchResults<object> fullResults = await textSearch.GetSearchResultsAsync(query, new() { Top = 4, Skip = 8 });
Console.WriteLine("\n--- Bing Web Page Results ---\n");
await foreach (BingWebPage result in fullResults.Results)
{
Console.WriteLine($"Name: {result.Name}");
Console.WriteLine($"Snippet: {result.Snippet}");
Console.WriteLine($"Url: {result.Url}");
Console.WriteLine($"DisplayUrl: {result.DisplayUrl}");
Console.WriteLine($"DateLastCrawled: {result.DateLastCrawled}");
WriteHorizontalRule();
}
}
/// <summary>
/// Show how to create a <see cref="BingTextSearch"/> with a custom mapper and use it to perform a text search.
/// </summary>
[Fact]
public async Task UsingBingTextSearchWithACustomMapperAsync()
{
// Create a logging handler to output HTTP requests and responses
LoggingHandler handler = new(new HttpClientHandler(), this.Output);
using HttpClient httpClient = new(handler);
// Create an ITextSearch instance using Bing search
var textSearch = new BingTextSearch(apiKey: TestConfiguration.Bing.ApiKey, options: new()
{
HttpClient = httpClient,
StringMapper = new TestTextSearchStringMapper(),
});
var query = "What is the Semantic Kernel?";
// Search with TextSearchResult textResult type
KernelSearchResults<string> stringResults = await textSearch.SearchAsync(query, new() { Top = 2, Skip = 0 });
Console.WriteLine("--- Serialized JSON Results ---");
await foreach (string result in stringResults.Results)
{
Console.WriteLine(result);
WriteHorizontalRule();
}
}
/// <summary>
/// Show how to create a <see cref="BingTextSearch"/> with a custom mapper and use it to perform a text search.
/// </summary>
[Fact]
public async Task UsingBingTextSearchWithASiteFilterAsync()
{
// Create a logging handler to output HTTP requests and responses
LoggingHandler handler = new(new HttpClientHandler(), this.Output);
using HttpClient httpClient = new(handler);
// Create an ITextSearch instance using Bing search
var textSearch = new BingTextSearch(apiKey: TestConfiguration.Bing.ApiKey, options: new()
{
HttpClient = httpClient,
StringMapper = new TestTextSearchStringMapper(),
});
var query = "What is the Semantic Kernel?";
// Search with TextSearchResult textResult type
TextSearchOptions searchOptions = new() { Top = 4, Skip = 0, Filter = new TextSearchFilter().Equality("site", "devblogs.microsoft.com") };
KernelSearchResults<TextSearchResult> textResults = await textSearch.GetTextSearchResultsAsync(query, searchOptions);
Console.WriteLine("--- Microsoft Developer Blogs Results ---");
await foreach (TextSearchResult result in textResults.Results)
{
Console.WriteLine(result.Link);
WriteHorizontalRule();
}
}
/// <summary>
/// Show how to use enhanced LINQ filtering with BingTextSearch for type-safe searches.
/// </summary>
[Fact]
public async Task UsingBingTextSearchWithLinqFilteringAsync()
{
// Create a logging handler to output HTTP requests and responses
LoggingHandler handler = new(new HttpClientHandler(), this.Output);
using HttpClient httpClient = new(handler);
// Create an ITextSearch<BingWebPage> instance for type-safe LINQ filtering
ITextSearch<BingWebPage> textSearch = new BingTextSearch(apiKey: TestConfiguration.Bing.ApiKey, options: new() { HttpClient = httpClient });
var query = "Semantic Kernel AI";
// Example 1: Filter by language (English only)
Console.WriteLine("——— Example 1: Language Filter (English) ———\n");
var languageOptions = new TextSearchOptions<BingWebPage>
{
Top = 2,
Filter = page => page.Language == "en"
};
var languageResults = await textSearch.SearchAsync(query, languageOptions);
await foreach (string result in languageResults.Results)
{
Console.WriteLine(result);
WriteHorizontalRule();
}
// Example 2: Filter by family-friendly content
Console.WriteLine("\n——— Example 2: Family Friendly Filter ———\n");
var familyFriendlyOptions = new TextSearchOptions<BingWebPage>
{
Top = 2,
Filter = page => page.IsFamilyFriendly == true
};
var familyFriendlyResults = await textSearch.SearchAsync(query, familyFriendlyOptions);
await foreach (string result in familyFriendlyResults.Results)
{
Console.WriteLine(result);
WriteHorizontalRule();
}
// Example 3: Compound AND filtering (language + family-friendly)
Console.WriteLine("\n——— Example 3: Compound Filter (English + Family Friendly) ———\n");
var compoundOptions = new TextSearchOptions<BingWebPage>
{
Top = 2,
Filter = page => page.Language == "en" && page.IsFamilyFriendly == true
};
var compoundResults = await textSearch.GetSearchResultsAsync(query, compoundOptions);
await foreach (BingWebPage page in compoundResults.Results)
{
Console.WriteLine($"Name: {page.Name}");
Console.WriteLine($"Snippet: {page.Snippet}");
Console.WriteLine($"Language: {page.Language}");
Console.WriteLine($"Family Friendly: {page.IsFamilyFriendly}");
WriteHorizontalRule();
}
// Example 4: Complex compound filtering with nullable checks
Console.WriteLine("\n——— Example 4: Complex Compound Filter (Language + Site + Family Friendly) ———\n");
var complexOptions = new TextSearchOptions<BingWebPage>
{
Top = 2,
Filter = page => page.Language == "en" &&
page.IsFamilyFriendly == true &&
page.DisplayUrl != null && page.DisplayUrl.Contains("microsoft")
};
var complexResults = await textSearch.GetSearchResultsAsync(query, complexOptions);
await foreach (BingWebPage page in complexResults.Results)
{
Console.WriteLine($"Name: {page.Name}");
Console.WriteLine($"Display URL: {page.DisplayUrl}");
Console.WriteLine($"Language: {page.Language}");
Console.WriteLine($"Family Friendly: {page.IsFamilyFriendly}");
WriteHorizontalRule();
}
}
#region private
/// <summary>
/// Test mapper which converts an arbitrary search result to a string using JSON serialization.
/// </summary>
private sealed class TestTextSearchStringMapper : ITextSearchStringMapper
{
/// <inheritdoc />
public string MapFromResultToString(object result)
{
return JsonSerializer.Serialize(result);
}
}
#endregion
}