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
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
ash / webui / help_app_ui / resources / browser_proxy.ts [blame]
// Copyright 2020 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import {stringToMojoString16} from './mojo_type_util.js';
import {String16} from 'chrome://resources/mojo/mojo/public/mojom/base/string16.mojom-webui.js';
import {Url} from 'chrome://resources/mojo/url/mojom/url.mojom-webui.js';
import {PageHandlerFactory, PageHandlerRemote} from './help_app_ui.mojom-webui.js';
import {Index} from './index.mojom-webui.js';
import {MessagePipe} from '//system_apps/message_pipe.js';
import {Message} from './message_types.js';
import {SearchConcept, SearchHandler} from './search.mojom-webui.js';
import {Content, ResponseStatus, Result} from './types.mojom-webui.js';
const helpApp = {
handler: new PageHandlerRemote(),
};
// Set up a page handler to talk to the browser process.
PageHandlerFactory.getRemote().createPageHandler(
helpApp.handler.$.bindNewPipeAndPassReceiver());
// Set up an index remote to talk to Local Search Service.
const indexRemote = Index.getRemote();
// Expose `indexRemote` on `window`, because it is accessed by a CrOS Tast test.
Object.assign(window, {indexRemote});
/**
* Talks to the search handler. Use for updating the content for launcher
* search.
*/
const searchHandlerRemote = SearchHandler.getRemote();
const GUEST_ORIGIN = 'chrome-untrusted://help-app';
const MAX_STRING_LEN = 9999;
const guestFrame = document.createElement('iframe');
guestFrame.src = `${GUEST_ORIGIN}${location.pathname}${location.search}`;
document.body.appendChild(guestFrame);
// Cached result of whether Launcher Search is enabled.
const isLauncherSearchEnabled =
helpApp.handler.isLauncherSearchEnabled().then(result => result.enabled);
/** Converts a string or object to url. */
function toUrl(url: string|object): Url {
// TODO(b/279132899): Figure out why `url` is an empty object when it should
// have been an empty string.
if (url === '' || typeof (url) !== 'string') {
return {url: ''};
}
return {url};
}
/** Converts string to String16. */
function toTruncatedString16(s: string): String16 {
return stringToMojoString16(truncate(s));
}
const TITLE_ID = 'title';
const BODY_ID = 'body';
const CATEGORY_ID = 'main-category';
const SUBCATEGORY_ID = 'subcategory';
const SUBHEADING_ID = 'subheading';
/**
* A pipe through which we can send messages to the guest frame.
* Use an undefined `target` to find the <iframe> automatically.
* Do not rethrow errors, since handlers installed here are expected to
* throw exceptions that are handled on the other side of the pipe (in the guest
* frame), not on this side.
*/
const guestMessagePipe = new MessagePipe(
'chrome-untrusted://help-app', /*target=*/ undefined,
/*rethrowErrors=*/ false);
guestMessagePipe.registerHandler(Message.OPEN_FEEDBACK_DIALOG, () => {
return helpApp.handler.openFeedbackDialog();
});
guestMessagePipe.registerHandler(
Message.SHOW_ON_DEVICE_APP_CONTROLS,
() => void helpApp.handler.showOnDeviceAppControls());
guestMessagePipe.registerHandler(Message.SHOW_PARENTAL_CONTROLS, () => void
helpApp.handler.showParentalControls());
guestMessagePipe.registerHandler(
Message.TRIGGER_WELCOME_TIP_CALL_TO_ACTION, (actionTypeId: number) => void
helpApp.handler.triggerWelcomeTipCallToAction(actionTypeId));
guestMessagePipe.registerHandler(
Message.ADD_OR_UPDATE_SEARCH_INDEX, async (data: SearchableItem[]) => {
const dataToSend = data.map(searchableItem => {
const contents: Content[] = [
{
id: TITLE_ID,
content: toTruncatedString16(searchableItem.title),
weight: 1.0,
},
{
id: CATEGORY_ID,
content: toTruncatedString16(searchableItem.mainCategoryName),
weight: 0.1,
},
];
if (searchableItem.subcategoryNames) {
for (let i = 0; i < searchableItem.subcategoryNames.length; ++i) {
const subcategoryName = searchableItem.subcategoryNames[i]!;
contents.push({
id: SUBCATEGORY_ID + i,
content: toTruncatedString16(subcategoryName),
weight: 0.1,
});
}
}
// If there are subheadings, use those instead of the body.
const subheadings = searchableItem.subheadings;
if (subheadings) {
for (let i = 0; i < subheadings.length; ++i) {
const subheading = subheadings[i];
if (!subheading) continue;
contents.push({
id: SUBHEADING_ID + i,
content: toTruncatedString16(subheading),
weight: 0.4,
});
}
} else if (searchableItem.body) {
contents.push({
id: BODY_ID,
content: toTruncatedString16(searchableItem.body),
weight: 0.2,
});
}
return {
id: searchableItem.id,
contents,
locale: searchableItem.locale,
};
});
return indexRemote.addOrUpdate(dataToSend);
});
guestMessagePipe.registerHandler(Message.CLEAR_SEARCH_INDEX, async () => {
return indexRemote.clearIndex();
});
guestMessagePipe.registerHandler(
Message.FIND_IN_SEARCH_INDEX,
async (dataFromApp: {query: string, maxResults: number}) => {
const response = await indexRemote.find(
toTruncatedString16(dataFromApp.query), dataFromApp.maxResults || 50);
if (response.status !== ResponseStatus.kSuccess || !response.results) {
return {results: null};
}
const searchResults: Result[] = (response.results);
// Sort results by decreasing score.
searchResults.sort((a, b) => b.score - a.score);
const results: SearchResult[] = searchResults.map(result => {
const titlePositions: Position[] = [];
const bodyPositions: Position[] = [];
// Id of the best subheading that appears in positions. We consider
// the subheading containing the most match positions to be the best.
// "" means no subheading positions found.
let bestSubheadingId = '';
/** Counts how many positions there are for each subheading id. */
const subheadingPosCounts: Record<string, number> = {};
// Note: result.positions is not sorted.
for (const position of result.positions) {
if (position.contentId === TITLE_ID) {
titlePositions.push(
{length: position.length, start: position.start});
} else if (position.contentId === BODY_ID) {
bodyPositions.push(
{length: position.length, start: position.start});
} else if (position.contentId.startsWith(SUBHEADING_ID)) {
// Update the subheadings's position count and check if it's the new
// best subheading.
const newCount = (subheadingPosCounts[position.contentId] || 0) + 1;
subheadingPosCounts[position.contentId] = newCount;
if (!bestSubheadingId ||
newCount > (subheadingPosCounts[bestSubheadingId] ?? 0)) {
bestSubheadingId = position.contentId;
}
}
}
// Use only the positions of the best subheading.
const subheadingPositions: Position[] = [];
if (bestSubheadingId) {
for (const position of result.positions) {
if (position.contentId === bestSubheadingId) {
subheadingPositions.push({
start: position.start,
length: position.length,
});
}
}
subheadingPositions.sort(compareByStart);
}
// Sort positions by start index.
titlePositions.sort(compareByStart);
bodyPositions.sort(compareByStart);
return {
id: result.id,
titlePositions,
bodyPositions,
subheadingIndex: bestSubheadingId ?
Number(bestSubheadingId.substring(SUBHEADING_ID.length)) : null,
subheadingPositions: bestSubheadingId ? subheadingPositions : null,
};
});
return {results};
},
);
guestMessagePipe.registerHandler(Message.CLOSE_BACKGROUND_PAGE, async () => {
// TODO(b/186180962): Add background page and test that it closes when done.
if (window.location.pathname !== '/background') {
return;
}
window.close();
return;
});
guestMessagePipe.registerHandler(
Message.UPDATE_LAUNCHER_SEARCH_INDEX,
async (message: LauncherSearchableItem[]) => {
if (!(await isLauncherSearchEnabled)) {
return;
}
const dataToSend: SearchConcept[] = message.map(
searchableItem => ({
id: truncate(searchableItem.id),
title: toTruncatedString16(searchableItem.title),
mainCategory: toTruncatedString16(searchableItem.mainCategoryName),
tags: searchableItem.tags.map(tag => toTruncatedString16(tag))
.filter(tag => tag.data.length > 0),
tagLocale: searchableItem.tagLocale || '',
urlPathWithParameters:
truncate(searchableItem.urlPathWithParameters),
locale: truncate(searchableItem.locale),
}));
// Filter out invalid items. No field can be empty except locale.
const dataFiltered = dataToSend.filter(item => {
const valid = item.id && item.title && item.mainCategory &&
item.tags.length > 0 && item.urlPathWithParameters;
// This is a google-internal histogram. If changing this, also change
// the corresponding histograms file.
if (!valid) {
(window as any).chrome.metricsPrivate
.recordSparseValueWithPersistentHash(
'Discover.LauncherSearch.InvalidConceptInUpdate', item.id);
}
return valid;
});
return searchHandlerRemote.update(dataFiltered);
},
);
guestMessagePipe.registerHandler(Message.LAUNCH_MICROSOFT_365_SETUP, () => void
helpApp.handler.launchMicrosoft365Setup());
guestMessagePipe.registerHandler(
Message.MAYBE_SHOW_RELEASE_NOTES_NOTIFICATION, () => void
helpApp.handler.maybeShowReleaseNotesNotification());
guestMessagePipe.registerHandler(Message.GET_DEVICE_INFO, async () => {
return (await helpApp.handler.getDeviceInfo()).deviceInfo;
});
guestMessagePipe.registerHandler(
Message.OPEN_SETTINGS,
(path: number) => void helpApp.handler.openSettings(path));
guestMessagePipe.registerHandler(
Message.OPEN_URL_IN_BROWSER_AND_TRIGGER_INSTALL_DIALOG,
(url: string | object) => {
helpApp.handler.openUrlInBrowserAndTriggerInstallDialog(toUrl(url));
},
);
guestMessagePipe.registerHandler(
Message.SET_HAS_COMPLETED_NEW_DEVICE_CHECKLIST,
() => void helpApp.handler.setHasCompletedNewDeviceChecklist());
guestMessagePipe.registerHandler(
Message.SET_HAS_VISITED_HOW_TO_PAGE,
() => void helpApp.handler.setHasVisitedHowToPage());
guestMessagePipe.registerHandler(
Message.OPEN_APP_MALL_PATH, ({path}: {path: string}) => {
window.open(`chrome://mall/${path}`);
});
/** Compare two positions by their start index. Use for sorting. */
function compareByStart(a: Position, b: Position): number {
return a.start - b.start;
}
/**
* Limits the maximum length of the input string. Converts non-strings into
* empty string.
*
* @param s Probably a string, but might not be.
*/
function truncate(s: any): string {
if (typeof s !== 'string') {
return '';
}
if (s.length <= MAX_STRING_LEN) {
return s;
}
return s.substring(0, MAX_STRING_LEN);
}
export const TEST_ONLY = {guestMessagePipe};