// Copyright 2015 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.


var properties =
    ["bla", "0", 1, Symbol(), {[Symbol.toPrimitive]() {return "a"}}];


function TestForwarding(handler, myDelete, shouldThrow) {
  var target = {};
  var proxy = new Proxy(target, handler);

  assertFalse(target.hasOwnProperty("doesnotexist"));
  assertTrue(myDelete(proxy, "doesnotexist"));

  for (p of properties) {
    target[p] = 42;
    assertTrue(myDelete(proxy, p));
    assertFalse(target.hasOwnProperty(p));
  }

  for (p of properties) {
    Object.defineProperty(target, p, {value: 42, configurable: false});
    if (shouldThrow) {
      assertThrows(() => myDelete(proxy, p), TypeError);
    } else {
      assertFalse(myDelete(proxy, p));
    }
    assertTrue(target.hasOwnProperty(p));
  }
};


(function () {
  // No trap.

  var handler = {};

  TestForwarding(handler,
      (o, p) => delete o[p], false);
  TestForwarding(handler,
      (o, p) => Reflect.deleteProperty(o, p), false);
  TestForwarding(handler,
      (o, p) => {"use strict"; return delete o[p]}, true);
  TestForwarding(handler,
      (o, p) => {"use strict"; return Reflect.deleteProperty(o, p)}, false);
})();


(function () {
  // "Undefined" trap.

  var handler = { deleteProperty: null };

  TestForwarding(handler,
      (o, p) => delete o[p], false);
  TestForwarding(handler,
      (o, p) => Reflect.deleteProperty(o, p), false);
  TestForwarding(handler,
      (o, p) => {"use strict"; return delete o[p]}, true);
  TestForwarding(handler,
      (o, p) => {"use strict"; return Reflect.deleteProperty(o, p)}, false);
})();


(function () {
  // Invalid trap.

  var target = {};
  var handler = { deleteProperty: true };
  var proxy = new Proxy(target, handler);

  assertThrows(() => delete proxy[0], TypeError);
  assertThrows(() => Reflect.deleteProperty(proxy, 0), TypeError);
})();


function TestTrappingTrueish(myDelete) {
  var handler = { deleteProperty() {return 42} };
  var target = {};
  var proxy = new Proxy(target, handler);

  // Trap returns trueish and target doesn't own property.
  for (p of properties) {
    assertTrue(myDelete(proxy, p));
  }

  // Trap returns trueish and target property is configurable.
  for (p of properties) {
    target[p] = 42;
    assertTrue(myDelete(proxy, p));
  }

  // Trap returns trueish but target property is not configurable.
  for (p of properties) {
    Object.defineProperty(target, p, {value: 42, configurable: false});
    assertThrows(() => myDelete(proxy, p), TypeError);
  }
};


TestTrappingTrueish(
    (o, p) => delete o[p]);
TestTrappingTrueish(
    (o, p) => Reflect.deleteProperty(o, p));
TestTrappingTrueish(
    (o, p) => {"use strict"; return delete o[p]});
TestTrappingTrueish(
    (o, p) => {"use strict"; return Reflect.deleteProperty(o, p)});


function TestTrappingTrueish2(myDelete) {
  var handler = {
      deleteProperty(target, p) {
          Object.defineProperty(target, p, {configurable: false});
          return 42
      }
  };
  var target = {};
  var proxy = new Proxy(target, handler);

  // Trap returns trueish but target property is not configurable.  In contrast
  // to above, here the target property was configurable before the trap call.
  for (p of properties) {
    target[p] = 42;
    assertThrows(() => myDelete(proxy, p), TypeError);
  }
};


TestTrappingTrueish2(
    (o, p) => delete o[p]);
TestTrappingTrueish2(
    (o, p) => Reflect.deleteProperty(o, p));
TestTrappingTrueish2(
    (o, p) => {"use strict"; return delete o[p]});
TestTrappingTrueish2(
    (o, p) => {"use strict"; return Reflect.deleteProperty(o, p)});


function TestTrappingFalsish(myDelete, shouldThrow) {
  var handler = { deleteProperty() {return ""} };
  var target = {};
  var proxy = new Proxy(target, handler);

  var properties =
      ["bla", "0", 1, Symbol(), {[Symbol.toPrimitive]() {return "a"}}];

  // Trap returns falsish and target doesn't own property.
  for (p of properties) {
    if (shouldThrow) {
      assertThrows(() => myDelete(proxy, p), TypeError);
    } else {
      assertFalse(myDelete(proxy, p));
    }
  }

  // Trap returns falsish and target property is configurable.
  for (p of properties) {
    target[p] = 42;
    if (shouldThrow) {
      assertThrows(() => myDelete(proxy, p), TypeError);
    } else {
      assertFalse(myDelete(proxy, p));
    }
  }

  // Trap returns falsish and target property is not configurable.
  for (p of properties) {
    Object.defineProperty(target, p, {value: 42, configurable: false});
    if (shouldThrow) {
      assertThrows(() => myDelete(proxy, p), TypeError);
    } else {
      assertFalse(myDelete(proxy, p));
    }
  }
};


TestTrappingFalsish(
    (o, p) => delete o[p], false);
TestTrappingFalsish(
    (o, p) => Reflect.deleteProperty(o, p), false);
TestTrappingFalsish(
    (o, p) => {"use strict"; return delete o[p]}, true);
TestTrappingFalsish(
    (o, p) => {"use strict"; return Reflect.deleteProperty(o, p)}, false);