facebookarchive / prepack

A JavaScript bundle optimizer.
http://prepack.io
Other
14.21k stars 424 forks source link

Constructors incorrectly forming conditionals when a throw exists in execution #2561

Open trueadm opened 6 years ago

trueadm commented 6 years ago

Here's a strange issue that I've been running into with the React Native bundle. Our module system initializes and when doing so, can run into cases where there's an abstract conditional that might throw. The throw seems to break the execution flow for function constructors, something that shouldn't happen.

Here's a repro:

function module() {
  var x = global.__abstract ? __abstract("boolean", "(true)") : true;

  // if the throw is remove, the string below gets inlined correctly
  if (!x) {
    throw new Error("An error");
  }

  var Component = function () {
    function Component() {
      this.someMethod = function() {
        return "This should be inlined!";
      };
    }

    return Component;
  }();

  function fn(x) {
    var instance = new Component();
    return instance.someMethod();
  }

  global.__optimize && __optimize(fn);

  return fn;
}

global.result = module();

When we get to var instance = new Component();. Prepack wrongly thinks that Component is a conditional abstract value of undefined | Constructor, because of the above throw. This results in bailing out and the code never inlining – something that break the React reconciler when dealing with React class components (we don't support conditional constructors).

hermanventer commented 6 years ago

This seems to be some kind of interaction between optimizing nested functions and joining the state of the local variables of module.

cblappert commented 6 years ago

Passes on master.

matthargett commented 6 years ago

if it passes on master, is there a missing test that should be added? or is the internal test difficult to reduce?