Skip to content

Commit

Permalink
Merge pull request #50361 from Expensify/beaman-fixLintMain
Browse files Browse the repository at this point in the history
  • Loading branch information
francoisl authored Oct 7, 2024
2 parents 1fb0d01 + 94fdd1a commit b9f8d39
Show file tree
Hide file tree
Showing 2 changed files with 168 additions and 167 deletions.
1 change: 1 addition & 0 deletions .eslintignore
Original file line number Diff line number Diff line change
Expand Up @@ -12,3 +12,4 @@ docs/assets/**
web/gtm.js
**/.expo/**
src/libs/SearchParser/searchParser.js
help/_scripts/**
334 changes: 167 additions & 167 deletions help/_scripts/generateSearchIndex.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,192 +5,192 @@ const cheerio = require('cheerio');

// Function to create the index from HTML files in '_site' directory
async function createIndex() {
console.log("Initializing FlexSearch index...");
let index;
try {
index = new FlexSearch.Document({
document: {
id: 'id',
index: ['content'], // Index on the content field
store: ['title', 'url', 'content'], // Store title, URL, and content
}
console.log('Initializing FlexSearch index...');
let index;
try {
index = new FlexSearch.Document({
document: {
id: 'id',
index: ['content'], // Index on the content field
store: ['title', 'url', 'content'], // Store title, URL, and content
},
});
console.log('FlexSearch index initialized successfully.');
} catch (error) {
console.error('Error initializing FlexSearch index:', error);
process.exit(1);
}

// Read HTML files from the '_site' directory
const dir = path.join(__dirname, '../_site');
console.log('Reading HTML files from directory:', dir);
let htmlFiles;
try {
htmlFiles = fs.readdirSync(dir).filter((file) => file.endsWith('.html'));
console.log('HTML files found:', htmlFiles);
} catch (error) {
console.error('Error reading directory:', error);
process.exit(1);
}

// Process each HTML file and add sections to the index
let indexId = 0;
htmlFiles.forEach((file, id) => {
console.log('Processing file:', file);
try {
const content = fs.readFileSync(path.join(dir, file), 'utf8');
const $ = cheerio.load(content);

// Extract sections with <h1>, <h2>, <h3> tags and add to index
$('h1, h2, h3').each((i, elem) => {
const title = $(elem).text().trim();
const sectionContent = $(elem).nextUntil('h1, h2, h3').text().trim();
const fullContent = `${title} ${sectionContent}`;

// Generate a URL for the section using the file path and a unique id based on the header
let sectionId;
if (elem.tagName === 'h1' && i === 0) {
// If it's the first <h1> tag, use the root URL (no fragment)
sectionId = '';
} else {
// Otherwise, generate an ID based on the element's position or existing ID
sectionId = $(elem).attr('id') || `${file}-section-${i}`;
$(elem).attr('id', sectionId); // Ensure the section has an id attribute
}
const sectionUrl = sectionId ? `${file}#${sectionId}` : `${file}#`;

console.log(`Found section: <${elem.tagName}> ${title} (${fullContent.length} characters)`);

// Add the section to the index with the section's URL, title, and content
index.add({
id: indexId++, // Give each a unique numerical index (see FlexSearch "best practices")
title,
url: sectionUrl,
content: fullContent,
});
console.log(`Index entry added for section in file ${file}`);
});
} catch (error) {
console.error(`Error processing file ${file}:`, error);
process.exit(1);
}
});
console.log("FlexSearch index initialized successfully.");
} catch (error) {
console.error("Error initializing FlexSearch index:", error);
process.exit(1);
}

// Read HTML files from the '_site' directory
const dir = path.join(__dirname, '../_site');
console.log("Reading HTML files from directory:", dir);
let htmlFiles;
try {
htmlFiles = fs.readdirSync(dir).filter(file => file.endsWith('.html'));
console.log("HTML files found:", htmlFiles);
} catch (error) {
console.error("Error reading directory:", error);
process.exit(1);
}

// Process each HTML file and add sections to the index
let indexId = 0;
htmlFiles.forEach((file, id) => {
console.log("Processing file:", file);

// Export the index as a JSON file
console.log('Exporting search index as JSON...');
let exportedIndex = {};
try {
const content = fs.readFileSync(path.join(dir, file), 'utf8');
const $ = cheerio.load(content);

// Extract sections with <h1>, <h2>, <h3> tags and add to index
$('h1, h2, h3').each((i, elem) => {
const title = $(elem).text().trim();
const sectionContent = $(elem).nextUntil('h1, h2, h3').text().trim();
const fullContent = `${title} ${sectionContent}`;

// Generate a URL for the section using the file path and a unique id based on the header
let sectionId;
if (elem.tagName === 'h1' && i === 0) {
// If it's the first <h1> tag, use the root URL (no fragment)
sectionId = '';
await index.export((key, data) => {
if (data) {
exportedIndex[key] = data;
}
});
if (Object.keys(exportedIndex).length === 0) {
console.warn('Warning: Exported index is empty.');
process.exit(1);
} else {
// Otherwise, generate an ID based on the element's position or existing ID
sectionId = $(elem).attr('id') || `${file}-section-${i}`;
$(elem).attr('id', sectionId); // Ensure the section has an id attribute
}
const sectionUrl = sectionId ? `${file}#${sectionId}` : `${file}#`;
console.log('Index exported successfully.');

console.log(`Found section: <${elem.tagName}> ${title} (${fullContent.length} characters)`);
// Write the exported index to a JSON file
const outputPath = path.join(dir, 'searchIndex.json');
console.log('Writing search index to JSON file:', outputPath);
const outputContent = JSON.stringify(exportedIndex, null, 2);

// Add the section to the index with the section's URL, title, and content
index.add({
id: indexId++, // Give each a unique numerical index (see FlexSearch "best practices")
title,
url: sectionUrl,
content: fullContent
});
console.log(`Index entry added for section in file ${file}`);
});
fs.writeFileSync(outputPath, outputContent);
console.log('Search index written to JSON file successfully.');
}
} catch (error) {
console.error(`Error processing file ${file}:`, error);
process.exit(1);
}
});

// Export the index as a JSON file
console.log("Exporting search index as JSON...");
let exportedIndex = {};
try {
await index.export((key, data) => {
if (data) {
exportedIndex[key] = data;
}
});
if (Object.keys(exportedIndex).length === 0) {
console.warn("Warning: Exported index is empty.");
process.exit(1);
} else {
console.log("Index exported successfully.");

// Write the exported index to a JSON file
const outputPath = path.join(dir, 'searchIndex.json');
console.log("Writing search index to JSON file:", outputPath);
const outputContent = JSON.stringify(exportedIndex, null, 2);

fs.writeFileSync(outputPath, outputContent);
console.log("Search index written to JSON file successfully.");
console.error('Error exporting search index:', error);
process.exit(1);
}
} catch (error) {
console.error("Error exporting search index:", error);
process.exit(1);
}
}

// Function to read and search the index with context and section title
async function searchIndex() {
console.log("\n--- Running Search Test ---");
const indexPath = path.join(__dirname, '../_site', 'searchIndex.json');
console.log("Loading search index from file:", indexPath);
let searchIndex;
try {
const indexFile = fs.readFileSync(indexPath, 'utf8');
searchIndex = JSON.parse(indexFile);
console.log("Search index loaded successfully. Number of entries:", Object.keys(searchIndex).length);
} catch (error) {
console.error("Error loading search index from file:", error);
process.exit(1);
}

// Initialize a new search index instance with loaded data
let index;
try {
index = new FlexSearch.Document({
document: {
id: 'id',
index: ['content'],
store: ['title', 'url', 'content'],
}
});
console.log('\n--- Running Search Test ---');
const indexPath = path.join(__dirname, '../_site', 'searchIndex.json');
console.log('Loading search index from file:', indexPath);
let searchIndex;
try {
const indexFile = fs.readFileSync(indexPath, 'utf8');
searchIndex = JSON.parse(indexFile);
console.log('Search index loaded successfully. Number of entries:', Object.keys(searchIndex).length);
} catch (error) {
console.error('Error loading search index from file:', error);
process.exit(1);
}

// Initialize a new search index instance with loaded data
let index;
try {
index = new FlexSearch.Document({
document: {
id: 'id',
index: ['content'],
store: ['title', 'url', 'content'],
},
});

// Import each key and corresponding data
for (const [key, data] of Object.entries(searchIndex)) {
await index.import(key, data);
}

// Import each key and corresponding data
for (const [key, data] of Object.entries(searchIndex)) {
await index.import(key, data);
console.log('Search index imported successfully.');
} catch (error) {
console.error('Error initializing or importing search index:', error);
process.exit(1);
}

console.log("Search index imported successfully.");
} catch (error) {
console.error("Error initializing or importing search index:", error);
process.exit(1);
}

// Perform a search test
let testKeyword = "superapp";
console.log(`Searching for the term '${testKeyword}' in the index...`);
try {
const results = await index.search({
query: testKeyword, // Do a partial case insensitive search
field: 'content'
});
if (results && results.length > 0) {
results.forEach(result => {
result.result.forEach(docId => {
const doc = index.store[docId];
if (doc && doc.content) { // Ensure doc and doc.content are defined
const content = doc.content;
const searchTermIndex = content.toLowerCase().indexOf(testKeyword);

if (searchTermIndex !== -1) {
// Extract context around the search term (30 characters before and after)
const contextBefore = content.substring(Math.max(0, searchTermIndex - 30), searchTermIndex);
const contextAfter = content.substring(searchTermIndex + testKeyword.length, Math.min(content.length, searchTermIndex + testKeyword.length + 30));

// Print the section title, URL, and context
console.log(`Section Title: ${doc.title}`);
console.log(`URL: ${doc.url}`);
console.log(`Context: ...${contextBefore}${testKeyword}${contextAfter}...`);
console.log(); // Add an empty line for better readability
} else {
console.warn(`Warning: Search term '${testKeyword}' not found in document ID: ${docId}`);
}
} else {
console.warn(`Warning: Document content not found for ID: ${docId}`);
}
// Perform a search test
let testKeyword = 'superapp';
console.log(`Searching for the term '${testKeyword}' in the index...`);
try {
const results = await index.search({
query: testKeyword, // Do a partial case insensitive search
field: 'content',
});
});
} else {
console.warn("Warning: No search results found.");
process.exit(1);
if (results && results.length > 0) {
results.forEach((result) => {
result.result.forEach((docId) => {
const doc = index.store[docId];
if (doc && doc.content) {
// Ensure doc and doc.content are defined
const content = doc.content;
const searchTermIndex = content.toLowerCase().indexOf(testKeyword);

if (searchTermIndex !== -1) {
// Extract context around the search term (30 characters before and after)
const contextBefore = content.substring(Math.max(0, searchTermIndex - 30), searchTermIndex);
const contextAfter = content.substring(searchTermIndex + testKeyword.length, Math.min(content.length, searchTermIndex + testKeyword.length + 30));

// Print the section title, URL, and context
console.log(`Section Title: ${doc.title}`);
console.log(`URL: ${doc.url}`);
console.log(`Context: ...${contextBefore}${testKeyword}${contextAfter}...`);
console.log(); // Add an empty line for better readability
} else {
console.warn(`Warning: Search term '${testKeyword}' not found in document ID: ${docId}`);
}
} else {
console.warn(`Warning: Document content not found for ID: ${docId}`);
}
});
});
} else {
console.warn('Warning: No search results found.');
process.exit(1);
}
} catch (error) {
console.error('Error searching the index:', error);
process.exit(1);
}
} catch (error) {
console.error("Error searching the index:", error);
process.exit(1);
}

process.exit(0);
process.exit(0);
}

// Run the main functions
(async () => {
await createIndex();
await searchIndex();
await createIndex();
await searchIndex();
})();

0 comments on commit b9f8d39

Please sign in to comment.