Skip to content
This repository was archived by the owner on Aug 22, 2023. It is now read-only.

Lazy require 'typescript' module. #72

Merged
merged 4 commits into from
Jan 31, 2019
Merged

Conversation

indexzero
Copy link
Contributor

Regression introduced by #70.

This is blocking. Currently any fresh install of a pure JavaScript @oclif project will die with:

Cannot find module 'typescript'
  1. No stack trace.
  2. No helpful message.
  3. Zip. Zilch. Zero.

Debugging aside, this is due to the fact that these JavaScript projects have no need to have typescript in the dependencies or devDependencies of their package.json.

Solution: since src/ts-node.js appears to exist to load TypeScript definitions for plugins and other modules consumed by a given CLI project using @oclif we only require typescript when we have confirmed that a tsconfig.json exists within that directory.

Not sure the right approach to test this, but hopefully we can land a hotfix and then discuss testability / debuggability separately.

@codecov
Copy link

codecov bot commented Jan 31, 2019

Codecov Report

Merging #72 into master will decrease coverage by 0.01%.
The diff coverage is 66.66%.

Impacted file tree graph

@@            Coverage Diff            @@
##           master     #72      +/-   ##
=========================================
- Coverage   68.61%   68.6%   -0.02%     
=========================================
  Files           7       7              
  Lines         427     430       +3     
  Branches      120     120              
=========================================
+ Hits          293     295       +2     
- Misses         85      86       +1     
  Partials       49      49
Impacted Files Coverage Δ
src/ts-node.ts 83.01% <66.66%> (-0.99%) ⬇️

Continue to review full report at Codecov.

Legend - Click here to learn more
Δ = absolute <relative> (impact), ø = not affected, ? = missing data
Powered by Codecov. Last update daf9ca4...49751a6. Read the comment docs.

src/ts-node.ts Outdated
let parseConfigFileTextToJson
try {
// Lazy-require 'typescript' to support JavaScript @oclif projects.
({parseConfigFileTextToJson} = require('typescript'))

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should probably just be done with dynamic import behaviors, something like the following:

async function loadTSConfig(root: string): TSConfig | undefined {
  const tsconfigPath = path.join(root, 'tsconfig.json')
  if (fs.existsSync(tsconfigPath)) {
    const { parseConfigFileTextToJson } = await import('typescript')

@jdx jdx merged commit 49ed0bd into oclif:master Jan 31, 2019
@indexzero indexzero deleted the lazy-typescript branch January 31, 2019 14:54

import Debug from './debug'
const debug = Debug()

let typescript: typeof import('typescript')

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since I am a dumb dummy, what is the purpose of dynamically importing typescript?

Couldn't you write this without the dynamic import? Also I haven't seen the let typescript: typeof import('typescript') syntax before. What is let typescript: doing?

let typescript;

try {
  typescript = require('typescript')
} catch (ex) {
  debug('Cannot find typescript', ex)
}

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You sir, are thinking like JavaScript! This was my original implementation, but TypeScript got very cranky with me since the type for let typescript could not be inferred automagically.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you for the explanation!

I am still not Typescript convinced, and I come from C# and Java background. I just haven't come across issues that I can see typescript solving that proper testing, prs, lint, and other process can't solve. I also can't see typescript solving the problems of those other areas, so they would still be required.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am a JavaScript person as well, but hey – open source means exploring new things 🤷‍♂️

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thinking further about your explanation, what does the type actually end up being from this:

let typescript: typeof import('typescript')

Wouldn't it be a promise? So it's saying that typescript is typeof promise, but it's not really a promise? Wouldn't it just be simpler to set the type to any or object or whatever type typescript would be?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

there are no types when the code is running, typescript types are only used at compile-time

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So I guess it's working because since typescript has to run to compile this source code, typescript is available at compile time in this scenario. So it just works. I wonder if this would work if the dependency was something other than typescript.

PS. Sorry for the chatty conversation, I fully respect your inboxes!

Copy link
Contributor

@trevor-scheer trevor-scheer Jan 31, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@blackdynamo - I agree with you. While it works, I think it's technically incorrect that typescript is being required at runtime, but not listed as a runtime dependency.

#72 (comment)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Okay, storytime. Back in the day there was a dream – a dream called optionalDependencies.

The current behavior of optionalDependencies is just a tiny fraction of what was discussed when it was being planned ... at least 6 years ago.

Behavior that would have covered this scenario: opt-in dependencies. What I mean by that is typescript is not actually a run-time dependency. It is a "nice to have" that only attempts to parse tsconfig.json file when present.

There are a lot of grey areas like this in production scenarios for package management. Lazy require is a well known technique to handle them which imho is appropriate here.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

TIL! Thanks for the further explanation 👍

Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants