Revert "Remove web-tree-sitter"

This reverts commit 5c98a3d4af.

# Conflicts:
#	package.json
This commit is contained in:
Saoud Rizwan
2024-07-31 07:56:42 -04:00
parent 45ee2c3e1c
commit d93a08af5d
5 changed files with 188 additions and 174 deletions

View File

@@ -66,7 +66,7 @@ const extensionConfig = {
sourcemap: !production,
logLevel: "silent",
plugins: [
//copyWasmFiles,
copyWasmFiles,
/* add to the end of plugins array */
esbuildProblemMatcherPlugin,
],

14
package-lock.json generated
View File

@@ -17,7 +17,9 @@
"globby": "^14.0.2",
"os-name": "^6.0.0",
"p-wait-for": "^5.0.2",
"serialize-error": "^11.0.3"
"serialize-error": "^11.0.3",
"tree-sitter-wasms": "^0.1.11",
"web-tree-sitter": "^0.22.6"
},
"devDependencies": {
"@types/diff": "^5.2.1",
@@ -5614,6 +5616,11 @@
"integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==",
"license": "MIT"
},
"node_modules/tree-sitter-wasms": {
"version": "0.1.11",
"resolved": "https://registry.npmjs.org/tree-sitter-wasms/-/tree-sitter-wasms-0.1.11.tgz",
"integrity": "sha512-26sE4+qoTi1CbzHdo9sHs9pRE/jXVFVRigSG/5TNAbwhSMVjHfMAg4UjmOhAFAIx5UxgoQuaURwqhm0SRNrpWA=="
},
"node_modules/ts-api-utils": {
"version": "1.3.0",
"resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.3.0.tgz",
@@ -5829,6 +5836,11 @@
"node": ">= 8"
}
},
"node_modules/web-tree-sitter": {
"version": "0.22.6",
"resolved": "https://registry.npmjs.org/web-tree-sitter/-/web-tree-sitter-0.22.6.tgz",
"integrity": "sha512-hS87TH71Zd6mGAmYCvlgxeGDjqd9GTeqXNqTT+u0Gs51uIozNIaaq/kUAbV/Zf56jb2ZOyG8BxZs2GG9wbLi6Q=="
},
"node_modules/webidl-conversions": {
"version": "3.0.1",
"resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz",

View File

@@ -2,7 +2,7 @@
"name": "claude-dev",
"displayName": "Claude Dev",
"description": "Autonomous software engineer right in your IDE, capable of reading/writing files, executing commands, and more with your permission every step of the way.",
"version": "1.0.64",
"version": "1.0.62",
"icon": "icon.png",
"engines": {
"vscode": "^1.84.0"
@@ -130,6 +130,8 @@
"globby": "^14.0.2",
"os-name": "^6.0.0",
"p-wait-for": "^5.0.2",
"serialize-error": "^11.0.3"
"serialize-error": "^11.0.3",
"tree-sitter-wasms": "^0.1.11",
"web-tree-sitter": "^0.22.6"
}
}

View File

@@ -1,7 +1,7 @@
import * as fs from "fs/promises"
import { globby } from "globby"
import * as path from "path"
//import { LanguageParser, loadRequiredLanguageParsers } from "./languageParser"
import { LanguageParser, loadRequiredLanguageParsers } from "./languageParser"
async function analyzeProject(dirPath: string): Promise<string> {
let result = ""
@@ -13,12 +13,12 @@ async function analyzeProject(dirPath: string): Promise<string> {
const { filesToParse, remainingFiles } = separateFiles(allFiles)
// Load only the necessary language parsers
//const languageParsers = await loadRequiredLanguageParsers(filesToParse)
const languageParsers = await loadRequiredLanguageParsers(filesToParse)
// Parse specific files we have language parsers for
const filesWithoutDefinitions: string[] = []
for (const file of filesToParse) {
const definitions = undefined //await parseFile(file, languageParsers)
const definitions = await parseFile(file, languageParsers)
if (definitions) {
if (!result) {
result += "# Source code definitions:\n\n"
@@ -119,68 +119,68 @@ This approach allows us to focus on the most relevant parts of the code (defined
- https://github.com/tree-sitter/tree-sitter/blob/master/lib/binding_web/test/helper.js
- https://tree-sitter.github.io/tree-sitter/code-navigation-systems
*/
// async function parseFile(filePath: string, languageParsers: LanguageParser): Promise<string | undefined> {
// const fileContent = await fs.readFile(filePath, "utf8")
// const ext = path.extname(filePath).toLowerCase().slice(1)
async function parseFile(filePath: string, languageParsers: LanguageParser): Promise<string | undefined> {
const fileContent = await fs.readFile(filePath, "utf8")
const ext = path.extname(filePath).toLowerCase().slice(1)
// const { parser, query } = languageParsers[ext] || {}
// if (!parser || !query) {
// return `Unsupported file type: ${filePath}`
const { parser, query } = languageParsers[ext] || {}
if (!parser || !query) {
return `Unsupported file type: ${filePath}`
}
let formattedOutput = ""
try {
// Parse the file content into an Abstract Syntax Tree (AST), a tree-like representation of the code
const tree = parser.parse(fileContent)
// Apply the query to the AST and get the captures
// Captures are specific parts of the AST that match our query patterns, each capture represents a node in the AST that we're interested in.
const captures = query.captures(tree.rootNode)
// Sort captures by their start position
captures.sort((a, b) => a.node.startPosition.row - b.node.startPosition.row)
// Split the file content into individual lines
const lines = fileContent.split("\n")
// Keep track of the last line we've processed
let lastLine = -1
captures.forEach((capture) => {
const { node, name } = capture
// Get the start and end lines of the current AST node
const startLine = node.startPosition.row
const endLine = node.endPosition.row
// Once we've retrieved the nodes we care about through the language query, we filter for lines with definition names only.
// name.startsWith("name.reference.") > refs can be used for ranking purposes, but we don't need them for the output
// previously we did `name.startsWith("name.definition.")` but this was too strict and excluded some relevant definitions
// Add separator if there's a gap between captures
if (lastLine !== -1 && startLine > lastLine + 1) {
formattedOutput += "|----\n"
}
// Only add the first line of the definition
// query captures includes the definition name and the definition implementation, but we only want the name (I found discrepencies in the naming structure for various languages, i.e. javascript names would be 'name' and typescript names would be 'name.definition)
if (name.includes("name") && lines[startLine]) {
formattedOutput += `${lines[startLine]}\n`
}
// Adds all the captured lines
// for (let i = startLine; i <= endLine; i++) {
// formattedOutput += `│${lines[i]}\n`
// }
//}
// let formattedOutput = ""
lastLine = endLine
})
} catch (error) {
console.log(`Error parsing file: ${error}\n`)
}
// try {
// // Parse the file content into an Abstract Syntax Tree (AST), a tree-like representation of the code
// const tree = parser.parse(fileContent)
// // Apply the query to the AST and get the captures
// // Captures are specific parts of the AST that match our query patterns, each capture represents a node in the AST that we're interested in.
// const captures = query.captures(tree.rootNode)
// // Sort captures by their start position
// captures.sort((a, b) => a.node.startPosition.row - b.node.startPosition.row)
// // Split the file content into individual lines
// const lines = fileContent.split("\n")
// // Keep track of the last line we've processed
// let lastLine = -1
// captures.forEach((capture) => {
// const { node, name } = capture
// // Get the start and end lines of the current AST node
// const startLine = node.startPosition.row
// const endLine = node.endPosition.row
// // Once we've retrieved the nodes we care about through the language query, we filter for lines with definition names only.
// // name.startsWith("name.reference.") > refs can be used for ranking purposes, but we don't need them for the output
// // previously we did `name.startsWith("name.definition.")` but this was too strict and excluded some relevant definitions
// // Add separator if there's a gap between captures
// if (lastLine !== -1 && startLine > lastLine + 1) {
// formattedOutput += "|----\n"
// }
// // Only add the first line of the definition
// // query captures includes the definition name and the definition implementation, but we only want the name (I found discrepencies in the naming structure for various languages, i.e. javascript names would be 'name' and typescript names would be 'name.definition)
// if (name.includes("name") && lines[startLine]) {
// formattedOutput += `│${lines[startLine]}\n`
// }
// // Adds all the captured lines
// // for (let i = startLine; i <= endLine; i++) {
// // formattedOutput += `│${lines[i]}\n`
// // }
// //}
// lastLine = endLine
// })
// } catch (error) {
// console.log(`Error parsing file: ${error}\n`)
// }
// if (formattedOutput.length > 0) {
// return `|----\n${formattedOutput}|----\n`
// }
// return undefined
// }
if (formattedOutput.length > 0) {
return `|----\n${formattedOutput}|----\n`
}
return undefined
}
export { analyzeProject }

View File

@@ -1,122 +1,122 @@
// import * as path from "path"
// import Parser from "web-tree-sitter"
// import {
// javascriptQuery,
// typescriptQuery,
// pythonQuery,
// rustQuery,
// goQuery,
// cppQuery,
// cQuery,
// csharpQuery,
// rubyQuery,
// javaQuery,
// phpQuery,
// swiftQuery,
// } from "./queries"
import * as path from "path"
import Parser from "web-tree-sitter"
import {
javascriptQuery,
typescriptQuery,
pythonQuery,
rustQuery,
goQuery,
cppQuery,
cQuery,
csharpQuery,
rubyQuery,
javaQuery,
phpQuery,
swiftQuery,
} from "./queries"
// export interface LanguageParser {
// [key: string]: {
// parser: Parser
// query: Parser.Query
// }
// }
export interface LanguageParser {
[key: string]: {
parser: Parser
query: Parser.Query
}
}
// async function loadLanguage(langName: string) {
// return await Parser.Language.load(path.join(__dirname, `tree-sitter-${langName}.wasm`))
// }
async function loadLanguage(langName: string) {
return await Parser.Language.load(path.join(__dirname, `tree-sitter-${langName}.wasm`))
}
// /*
// Using node bindings for tree-sitter is problematic in vscode extensions
// because of incompatibility with electron. Going the .wasm route has the
// advantage of not having to build for multiple architectures.
/*
Using node bindings for tree-sitter is problematic in vscode extensions
because of incompatibility with electron. Going the .wasm route has the
advantage of not having to build for multiple architectures.
// We use web-tree-sitter and tree-sitter-wasms which provides auto-updating prebuilt WASM binaries for tree-sitter's language parsers.
We use web-tree-sitter and tree-sitter-wasms which provides auto-updating prebuilt WASM binaries for tree-sitter's language parsers.
// This function loads WASM modules for relevant language parsers based on input files:
// 1. Extracts unique file extensions
// 2. Maps extensions to language names
// 3. Loads corresponding WASM files (containing grammar rules)
// 4. Uses WASM modules to initialize tree-sitter parsers
This function loads WASM modules for relevant language parsers based on input files:
1. Extracts unique file extensions
2. Maps extensions to language names
3. Loads corresponding WASM files (containing grammar rules)
4. Uses WASM modules to initialize tree-sitter parsers
// This approach optimizes performance by loading only necessary parsers once for all relevant files.
This approach optimizes performance by loading only necessary parsers once for all relevant files.
// Sources:
// - https://github.com/tree-sitter/node-tree-sitter/issues/169
// - https://github.com/tree-sitter/node-tree-sitter/issues/168
// - https://github.com/Gregoor/tree-sitter-wasms/blob/main/README.md
// - https://github.com/tree-sitter/tree-sitter/blob/master/lib/binding_web/README.md
// - https://github.com/tree-sitter/tree-sitter/blob/master/lib/binding_web/test/query-test.js
// */
// export async function loadRequiredLanguageParsers(filesToParse: string[]): Promise<LanguageParser> {
// await Parser.init()
// const extensionsToLoad = new Set(filesToParse.map((file) => path.extname(file).toLowerCase().slice(1)))
// const parsers: LanguageParser = {}
// for (const ext of extensionsToLoad) {
// let language: Parser.Language
// let query: Parser.Query
// switch (ext) {
// case "js":
// case "jsx":
// language = await loadLanguage("javascript")
// query = language.query(javascriptQuery)
// break
// case "ts":
// language = await loadLanguage("typescript")
// query = language.query(typescriptQuery)
// break
// case "tsx":
// language = await loadLanguage("tsx")
// query = language.query(typescriptQuery)
// break
// case "py":
// language = await loadLanguage("python")
// query = language.query(pythonQuery)
// break
// case "rs":
// language = await loadLanguage("rust")
// query = language.query(rustQuery)
// break
// case "go":
// language = await loadLanguage("go")
// query = language.query(goQuery)
// break
// case "cpp":
// case "hpp":
// language = await loadLanguage("cpp")
// query = language.query(cppQuery)
// break
// case "c":
// case "h":
// language = await loadLanguage("c")
// query = language.query(cQuery)
// break
// case "cs":
// language = await loadLanguage("c_sharp")
// query = language.query(csharpQuery)
// break
// case "rb":
// language = await loadLanguage("ruby")
// query = language.query(rubyQuery)
// break
// case "java":
// language = await loadLanguage("java")
// query = language.query(javaQuery)
// break
// case "php":
// language = await loadLanguage("php")
// query = language.query(phpQuery)
// break
// case "swift":
// language = await loadLanguage("swift")
// query = language.query(swiftQuery)
// break
// default:
// throw new Error(`Unsupported language: ${ext}`)
// }
// const parser = new Parser()
// parser.setLanguage(language)
// parsers[ext] = { parser, query }
// }
// return parsers
// }
Sources:
- https://github.com/tree-sitter/node-tree-sitter/issues/169
- https://github.com/tree-sitter/node-tree-sitter/issues/168
- https://github.com/Gregoor/tree-sitter-wasms/blob/main/README.md
- https://github.com/tree-sitter/tree-sitter/blob/master/lib/binding_web/README.md
- https://github.com/tree-sitter/tree-sitter/blob/master/lib/binding_web/test/query-test.js
*/
export async function loadRequiredLanguageParsers(filesToParse: string[]): Promise<LanguageParser> {
await Parser.init()
const extensionsToLoad = new Set(filesToParse.map((file) => path.extname(file).toLowerCase().slice(1)))
const parsers: LanguageParser = {}
for (const ext of extensionsToLoad) {
let language: Parser.Language
let query: Parser.Query
switch (ext) {
case "js":
case "jsx":
language = await loadLanguage("javascript")
query = language.query(javascriptQuery)
break
case "ts":
language = await loadLanguage("typescript")
query = language.query(typescriptQuery)
break
case "tsx":
language = await loadLanguage("tsx")
query = language.query(typescriptQuery)
break
case "py":
language = await loadLanguage("python")
query = language.query(pythonQuery)
break
case "rs":
language = await loadLanguage("rust")
query = language.query(rustQuery)
break
case "go":
language = await loadLanguage("go")
query = language.query(goQuery)
break
case "cpp":
case "hpp":
language = await loadLanguage("cpp")
query = language.query(cppQuery)
break
case "c":
case "h":
language = await loadLanguage("c")
query = language.query(cQuery)
break
case "cs":
language = await loadLanguage("c_sharp")
query = language.query(csharpQuery)
break
case "rb":
language = await loadLanguage("ruby")
query = language.query(rubyQuery)
break
case "java":
language = await loadLanguage("java")
query = language.query(javaQuery)
break
case "php":
language = await loadLanguage("php")
query = language.query(phpQuery)
break
case "swift":
language = await loadLanguage("swift")
query = language.query(swiftQuery)
break
default:
throw new Error(`Unsupported language: ${ext}`)
}
const parser = new Parser()
parser.setLanguage(language)
parsers[ext] = { parser, query }
}
return parsers
}