diff options
Diffstat (limited to 'extension/src/languages')
| -rw-r--r-- | extension/src/languages/index.d.ts | 13 | ||||
| -rw-r--r-- | extension/src/languages/index.ts | 19 | ||||
| -rw-r--r-- | extension/src/languages/javascript/index.ts | 16 | ||||
| -rw-r--r-- | extension/src/languages/notImplemented.ts | 10 | ||||
| -rw-r--r-- | extension/src/languages/python/index.ts | 74 | 
5 files changed, 132 insertions, 0 deletions
| diff --git a/extension/src/languages/index.d.ts b/extension/src/languages/index.d.ts new file mode 100644 index 00000000..be7ddfbc --- /dev/null +++ b/extension/src/languages/index.d.ts @@ -0,0 +1,13 @@ +export interface LanguageLibrary { +  language: string; +  fileExtensions: string[]; +  parseFirstStacktrace: (stdout: string) => string | undefined; +  lineIsFunctionDef: (line: string) => boolean; +  parseFunctionDefForName: (line: string) => string; +  lineIsComment: (line: string) => boolean; +  writeImport: ( +    sourcePath: string, +    pathToImport: string, +    namesToImport?: string[] | undefined +  ) => string; +} diff --git a/extension/src/languages/index.ts b/extension/src/languages/index.ts new file mode 100644 index 00000000..31d73a0b --- /dev/null +++ b/extension/src/languages/index.ts @@ -0,0 +1,19 @@ +import pythonLanguageLibrary from "./python"; +import javascriptLanguageLibrary from "./javascript"; +import { LanguageLibrary } from "./index.d"; + +export const languageLibraries: LanguageLibrary[] = [ +  pythonLanguageLibrary, +  javascriptLanguageLibrary, +]; + +export function getLanguageLibrary(filepath: string): LanguageLibrary { +  for (let languageLibrary of languageLibraries) { +    for (let fileExtension of languageLibrary.fileExtensions) { +      if (filepath.endsWith(fileExtension)) { +        return languageLibrary; +      } +    } +  } +  throw new Error(`No language library found for file ${filepath}`); +} diff --git a/extension/src/languages/javascript/index.ts b/extension/src/languages/javascript/index.ts new file mode 100644 index 00000000..1c21a2fc --- /dev/null +++ b/extension/src/languages/javascript/index.ts @@ -0,0 +1,16 @@ +import { LanguageLibrary } from "../index.d"; +import { notImplemented } from "../notImplemented"; + +const NI = (propertyName: string) => notImplemented(propertyName, "javascript"); + +const javascriptLangaugeLibrary: LanguageLibrary = { +  language: "javascript", +  fileExtensions: [".js", ".jsx", ".ts", ".tsx"], +  parseFirstStacktrace: NI("parseFirstStacktrace"), +  lineIsFunctionDef: NI("lineIsFunctionDef"), +  parseFunctionDefForName: NI("parseFunctionDefForName"), +  lineIsComment: NI("lineIsComment"), +  writeImport: NI("writeImport"), +}; + +export default javascriptLangaugeLibrary; diff --git a/extension/src/languages/notImplemented.ts b/extension/src/languages/notImplemented.ts new file mode 100644 index 00000000..bbba2382 --- /dev/null +++ b/extension/src/languages/notImplemented.ts @@ -0,0 +1,10 @@ +export function notImplemented( +  propertyName: string, +  langauge: string +): (...args: any[]) => never { +  return (...args: any[]) => { +    throw new Error( +      `Property ${propertyName} not implemented for language ${langauge}.` +    ); +  }; +} diff --git a/extension/src/languages/python/index.ts b/extension/src/languages/python/index.ts new file mode 100644 index 00000000..50282b45 --- /dev/null +++ b/extension/src/languages/python/index.ts @@ -0,0 +1,74 @@ +import path = require("path"); +import { LanguageLibrary } from "../index.d"; + +const tracebackStart = "Traceback (most recent call last):"; +const tracebackEnd = (buf: string): string | undefined => { +  let lines = buf +    .split("\n") +    .filter((line: string) => line.trim() !== "~~^~~") +    .filter((line: string) => line.trim() !== ""); +  for (let i = 0; i < lines.length; i++) { +    if ( +      lines[i].startsWith("  File") && +      i + 2 < lines.length && +      lines[i + 2][0] !== " " +    ) { +      return lines.slice(0, i + 3).join("\n"); +    } +  } +  return undefined; +}; + +function parseFirstStacktrace(stdout: string): string | undefined { +  let startIdx = stdout.indexOf(tracebackStart); +  if (startIdx < 0) return undefined; +  stdout = stdout.substring(startIdx); +  return tracebackEnd(stdout); +} + +function lineIsFunctionDef(line: string): boolean { +  return line.startsWith("def "); +} + +function parseFunctionDefForName(line: string): string { +  return line.split("def ")[1].split("(")[0]; +} + +function lineIsComment(line: string): boolean { +  return line.trim().startsWith("#"); +} + +function writeImport( +  sourcePath: string, +  pathToImport: string, +  namesToImport: string[] | undefined = undefined +): string { +  let segs = path.relative(sourcePath, pathToImport).split(path.sep); +  let importFrom = ""; +  for (let seg of segs) { +    if (seg === "..") { +      importFrom = "." + importFrom; +    } else { +      if (!importFrom.endsWith(".")) { +        importFrom += "."; +      } +      importFrom += seg.split(".").slice(0, -1).join("."); +    } +  } + +  return `from ${importFrom} import ${ +    namesToImport ? namesToImport.join(", ") : "*" +  }`; +} + +const pythonLangaugeLibrary: LanguageLibrary = { +  language: "python", +  fileExtensions: [".py"], +  parseFirstStacktrace, +  lineIsFunctionDef, +  parseFunctionDefForName, +  lineIsComment, +  writeImport, +}; + +export default pythonLangaugeLibrary; | 
