From 13b63c5486a9f5f21b371ad29b89898e71cf5132 Mon Sep 17 00:00:00 2001
From: Mark Miller <erights@gmail.com>
Date: Fri, 30 Sep 2011 08:24:38 -0400
Subject: [PATCH] test262 console runner working!

---
 test/harness/ed.js                 |  22 +--
 test/harness/gs.js                 |   2 +-
 test/harness/sta.js                |  67 +++----
 test/harness/sth.js                |  46 ++---
 tools/converter/convert.py         | 174 ------------------
 tools/packaging/parseTestRecord.py |  65 +++++++
 tools/{ => packaging}/test262.py   | 280 ++++++++++-------------------
 7 files changed, 231 insertions(+), 425 deletions(-)
 delete mode 100644 tools/converter/convert.py
 create mode 100644 tools/packaging/parseTestRecord.py
 rename tools/{ => packaging}/test262.py (62%)
 mode change 100644 => 100755

diff --git a/test/harness/ed.js b/test/harness/ed.js
index e2a27c6a3c..06c8ab07d7 100644
--- a/test/harness/ed.js
+++ b/test/harness/ed.js
@@ -1,14 +1,14 @@
-/// Copyright (c) 2011 Microsoft Corporation 
-/// 
+/// Copyright (c) 2011 Microsoft Corporation
+///
 /// Redistribution and use in source and binary forms, with or without modification, are permitted provided
-/// that the following conditions are met: 
+/// that the following conditions are met:
 ///    * Redistributions of source code must retain the above copyright notice, this list of conditions and
-///      the following disclaimer. 
-///    * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and 
-///      the following disclaimer in the documentation and/or other materials provided with the distribution.  
+///      the following disclaimer.
+///    * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and
+///      the following disclaimer in the documentation and/or other materials provided with the distribution.
 ///    * Neither the name of Microsoft nor the names of its contributors may be used to
 ///      endorse or promote products derived from this software without specific prior written permission.
-/// 
+///
 /// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR
 /// IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 /// FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
@@ -16,13 +16,13 @@
 /// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 /// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 /// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
-/// ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
+/// ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 
 //Error Detector
 if (this.window!==undefined) {  //for console support
-    window.onerror = function(errorMsg, url, lineNumber) {
-        window.iframeError = errorMsg;
-    }
+    this.window.onerror = function(errorMsg, url, lineNumber) {
+        this.window.iframeError = errorMsg;
+    };
 }
 
 //This doesn't work with early errors in current versions of Opera
diff --git a/test/harness/gs.js b/test/harness/gs.js
index bd9a66cd52..a488bfc5a3 100644
--- a/test/harness/gs.js
+++ b/test/harness/gs.js
@@ -1,4 +1,4 @@
-/// Copyright (c) 2011 Microsoft Corporation
+/// Copyright (c) 2011 Microsoft Corporation
 ///
 /// Redistribution and use in source and binary forms, with or without modification, are permitted provided
 /// that the following conditions are met:
diff --git a/test/harness/sta.js b/test/harness/sta.js
index c25198cb5f..944729b6a9 100644
--- a/test/harness/sta.js
+++ b/test/harness/sta.js
@@ -1,4 +1,4 @@
-/// Copyright (c) 2009 Microsoft Corporation
+/// Copyright (c) 2009 Microsoft Corporation
 ///
 /// Redistribution and use in source and binary forms, with or without modification, are permitted provided
 /// that the following conditions are met:
@@ -335,8 +335,8 @@ function getPrecision(num) {
     //TODO: Create a table of prec's,
     //      because using Math for testing Math isn't that correct.
 
-    log2num = Math.log(Math.abs(num)) / Math.LN2;
-    pernum = Math.ceil(log2num);
+    var log2num = Math.log(Math.abs(num)) / Math.LN2;
+    var pernum = Math.ceil(log2num);
     return (2 * Math.pow(2, -52 + pernum));
     //return(0);
 }
@@ -364,7 +364,7 @@ function isEqual(num1, num2) {
 // This code is governed by the BSD license found in the LICENSE file.
 
 function ToInteger(p) {
-    x = Number(p);
+    var x = Number(p);
 
     if (isNaN(x)) {
         return +0;
@@ -437,7 +437,7 @@ var $LocalTZ,
             current = new Date(current.getTime() + 1);
         }
         return current;
-    }
+    };
 
     var juneDate = new Date(2000, 5, 20, 0, 0, 0, 0);
     var decemberDate = new Date(2000, 11, 20, 0, 0, 0, 0);
@@ -446,7 +446,7 @@ var $LocalTZ,
     var isSouthernHemisphere = (juneOffset > decemberOffset);
     var winterTime = isSouthernHemisphere ? juneDate : decemberDate;
     var summerTime = isSouthernHemisphere ? decemberDate : juneDate;
-    
+
     var dstStart = findNearestDateBefore(winterTime, function (date) {
         return date.getTimezoneOffset() == summerTime.getTimezoneOffset();
     });
@@ -454,7 +454,7 @@ var $LocalTZ,
     $DST_start_sunday = dstStart.getDate() > 15 ? '"last"' : '"first"';
     $DST_start_hour = dstStart.getHours();
     $DST_start_minutes = dstStart.getMinutes();
-      
+
     var dstEnd = findNearestDateBefore(summerTime, function (date) {
         return date.getTimezoneOffset() == winterTime.getTimezoneOffset();
     });
@@ -462,7 +462,7 @@ var $LocalTZ,
     $DST_end_sunday = dstEnd.getDate() > 15 ? '"last"' : '"first"';
     $DST_end_hour = dstEnd.getHours();
     $DST_end_minutes = dstEnd.getMinutes();
-    
+
     return;
 })();
 
@@ -503,10 +503,10 @@ function YearFromTime(t) {
   t = Number(t);
   var sign = ( t < 0 ) ? -1 : 1;
   var year = ( sign < 0 ) ? 1969 : 1970;
-  
+
   for(var time = 0;;year += sign){
     time = TimeFromYear(year);
-    
+
     if(sign > 0 && time > t){
       year -= sign;
       break;
@@ -517,11 +517,11 @@ function YearFromTime(t) {
   };
   return year;
 }
-  
+
 function InLeapYear(t){
   if(DaysInYear(YearFromTime(t)) == 365)
     return 0;
-  
+
   if(DaysInYear(YearFromTime(t)) == 366)
     return 1;
 }
@@ -584,7 +584,7 @@ var LocalTZA = $LocalTZ*msPerHour;
 
 function DaysInMonth(m, leap) {
   m = m%12;
-  
+
   //April, June, Sept, Nov
   if(m == 3 || m == 5 || m == 8 || m == 10 ) {
     return 30;
@@ -601,13 +601,14 @@ function DaysInMonth(m, leap) {
 
 function GetSundayInMonth(t, m, count){
     var year = YearFromTime(t);
-    
+    var tempDate;
+
     if (count==='"first"') {
         for (var d=1; d <= DaysInMonth(m, InLeapYear(t)); d++) {
             tempDate = new Date(year, m, d);
             if (tempDate.getDay()===0) {
                 return tempDate.valueOf();
-            }         
+            }
         }
     } else if(count==='"last"') {
         for (var d=DaysInMonth(m, InLeapYear(t)); d>0; d--) {
@@ -624,7 +625,7 @@ function GetSundayInMonth(t, m, count){
   var year = YearFromTime(t);
   var leap = InLeapYear(t);
   var day = 0;
-  
+
   if(m >= 1) day += DaysInMonth(0, leap);
   if(m >= 2) day += DaysInMonth(1, leap);
   if(m >= 3) day += DaysInMonth(2, leap);
@@ -636,25 +637,25 @@ function GetSundayInMonth(t, m, count){
   if(m >= 9) day += DaysInMonth(8, leap);
   if(m >= 10) day += DaysInMonth(9, leap);
   if(m >= 11) day += DaysInMonth(10, leap);
-  
+
   var month_start = TimeFromYear(year)+day*msPerDay;
   var sunday = 0;
-  
+
   if(count === "last"){
-    for(var last_sunday = month_start+DaysInMonth(m, leap)*msPerDay; 
+    for(var last_sunday = month_start+DaysInMonth(m, leap)*msPerDay;
       WeekDay(last_sunday)>0;
       last_sunday -= msPerDay
     ){};
     sunday = last_sunday;
   }
   else {
-    for(var first_sunday = month_start; 
+    for(var first_sunday = month_start;
       WeekDay(first_sunday)>0;
       first_sunday += msPerDay
     ){};
     sunday = first_sunday+7*msPerDay*(count-1);
   }
-  
+
   return sunday;
 }*/
 
@@ -662,11 +663,11 @@ function DaylightSavingTA(t) {
 //  t = t-LocalTZA;
 
   var DST_start = GetSundayInMonth(t, $DST_start_month, $DST_start_sunday) +
-                  $DST_start_hour*msPerHour + 
+                  $DST_start_hour*msPerHour +
                   $DST_start_minutes*msPerMinute;
-                  
+
   var k = new Date(DST_start);
-  
+
   var DST_end   = GetSundayInMonth(t, $DST_end_month, $DST_end_sunday) +
                   $DST_end_hour*msPerHour +
                   $DST_end_minutes*msPerMinute;
@@ -770,7 +771,7 @@ function MakeDate( day, time ) {
   if(!isFinite(day) || !isFinite(time)) {
     return Number.NaN;
   }
-  
+
   return day*msPerDay+time;
 }
 
@@ -803,20 +804,20 @@ function ConstructDate(year, month, date, hours, minutes, seconds, ms){
   var r1 = Number(year);
   var r2 = Number(month);
   var r3 = ((date && arguments.length > 2) ? Number(date) : 1);
-  var r4 = ((hours && arguments.length > 3) ? Number(hours) : 0);   
-  var r5 = ((minutes && arguments.length > 4) ? Number(minutes) : 0);   
-  var r6 = ((seconds && arguments.length > 5) ? Number(seconds) : 0);   
+  var r4 = ((hours && arguments.length > 3) ? Number(hours) : 0);
+  var r5 = ((minutes && arguments.length > 4) ? Number(minutes) : 0);
+  var r6 = ((seconds && arguments.length > 5) ? Number(seconds) : 0);
   var r7 = ((ms && arguments.length > 6) ? Number(ms) : 0);
-  
+
   var r8 = r1;
-  
+
   if(!isNaN(r1) && (0 <= ToInteger(r1)) && (ToInteger(r1) <= 99))
     r8 = 1900+r1;
-  
+
   var r9 = MakeDay(r8, r2, r3);
   var r10 = MakeTime(r4, r5, r6, r7);
   var r11 = MakeDate(r9, r10);
-  
+
   return TimeClip(UTC(r11));
 }
 
@@ -905,6 +906,6 @@ return attribs
 //--Test case registration-----------------------------------------------------
 function runTestCase(testcase) {
     if (testcase() !== true) {
-        $ERROR("Test case returned non-true value!")
+        $ERROR("Test case returned non-true value!");
     }
 }
diff --git a/test/harness/sth.js b/test/harness/sth.js
index 8d2658d868..491eeedf5c 100644
--- a/test/harness/sth.js
+++ b/test/harness/sth.js
@@ -43,21 +43,21 @@ function BrowserRunner() {
         globalScopeContents,
         harnessDir = "harness/";
 
-    $.ajax({async: false, 
-            dataType: "text", 
-            success: function(data){errorDetectorFileContents = data;}, 
+    $.ajax({async: false,
+            dataType: "text",
+            success: function(data){errorDetectorFileContents = data;},
             url:harnessDir+"ed.js"});
-            
-    $.ajax({async: false, 
-            dataType: "text", 
-            success: function(data){simpleTestAPIContents = data;}, 
+
+    $.ajax({async: false,
+            dataType: "text",
+            success: function(data){simpleTestAPIContents = data;},
             url:harnessDir+"sta.js"});
-    
-    $.ajax({async: false, 
-            dataType: "text", 
-            success: function(data){globalScopeContents = data;}, 
+
+    $.ajax({async: false,
+            dataType: "text",
+            success: function(data){globalScopeContents = data;},
             url:harnessDir+"gs.js"});
-            
+
     /* Called by the child window to notify that the test has
      * finished. This function call is put in a separate script block
      * at the end of the page so errors in the test script block
@@ -180,8 +180,8 @@ function BrowserRunner() {
         idoc.writeln(globalScopeContents);
         idoc.writeln("</script>");
         idoc.close();
-    }
-    
+    };
+
     //--Helper functions-------------------------------------------------------
     this.convertForEval = function(txt) {
         txt = txt.replace(/\\/g,"\\\\");
@@ -190,7 +190,7 @@ function BrowserRunner() {
         txt = txt.replace(/\r/g,"\\r");
         txt = txt.replace(/\n/g,"\\n");
         return txt;
-    }
+    };
 }
 
 /* Loads tests from the sections specified in testcases.json.
@@ -264,7 +264,7 @@ function TestLoader() {
 
     function getIdFromPath (path) {
         //path is of the form "a/b/c.js"
-        
+
         var id = path.split("/");
         //id is now of the form ["a", "b", "c.js"];
 
@@ -307,9 +307,9 @@ function TestLoader() {
         currentTestIndex = 0;
         testGroupIndex = 0;
     };
-    
-    
-    
+
+
+
 }
 
 /* Controls test generation and running, and sends results to the presenter. */
@@ -326,8 +326,8 @@ function Controller() {
     this.implementerHook = {
         //Adds a test result
         addTestResult: function (test) { },
-    
-        //Called whenever all tests have finished running.  Provided with the 
+
+        //Called whenever all tests have finished running.  Provided with the
         //elapsed time in milliseconds.
         finished: function(elapsed) { }
     };
@@ -337,7 +337,7 @@ function Controller() {
         try {
             controller.implementerHook.addTestResult(test);
         } catch(e) { /*no-op*/}
-        
+
         if(state === 'running')
             setTimeout(loader.getNextTest, 10);
     };
@@ -366,7 +366,7 @@ function Controller() {
         try {
             controller.implementerHook.finished(elapsed);
         } catch(e) { /*no-op*/}
-    }
+    };
 
     this.start = function() {
         state = 'running';
diff --git a/tools/converter/convert.py b/tools/converter/convert.py
deleted file mode 100644
index f4171b9fe5..0000000000
--- a/tools/converter/convert.py
+++ /dev/null
@@ -1,174 +0,0 @@
-#!/usr/bin/env python
-
-# Copyright 2011 by Google, Inc.  All rights reserved.
-# This code is governed by the BSD license found in the LICENSE file.
-
-# Follows convert.js as closely as possible. So to minimize
-# divergence, see convert.js for doc-comments that are missing here.
-
-
-import logging
-import optparse
-import os
-from os import path
-import platform
-import re
-import subprocess
-import sys
-import tempfile
-import time
-
-# from TestCasePackagerConfig import *
-
-headerPattern = r"(?:(?:\/\/.*)?\s*\n)*"
-captureCommentPattern = r"\/\*\*?((?:\s|\S)*?)\*\/\s*\n"
-anyPattern = r"(?:\s|\S)*"
-blanksPattern = r"(?:\s|\n)*"
-
-# only until we've done our last conversion from current sputnik
-# format to canonical test262 format
-captureStrictPattern = r"\s*('use strict'|\"use strict\");"
-
-# Should match anything
-testEnvelopePattern = r"^(" + headerPattern +
-                      r")(?:" + captureCommentPattern +
-                      r")?(?:" + captureStrictPattern +
-                      r")?(" + anyPattern +
-                      r")$"
-
-registerPattern = r"^(" + anyPattern + r"?)(" +
-                  r"ES5Harness\.registerTest\s*\(\s*\{" + anyPattern +
-                  r"\}\s*\)" + r")" +
-                  r"\s*;?(?:\s|\n)*$"
-
-Matches a named function. Captures both the name and the body.
-captureFuncNameBodyPattern = r"^function\s+(\w*)\(\s*\)\s*\{" +
-                             r"(" + anyPattern + r")" +
-                             r";?" + blanksPattern +
-                             r"\}$"
-
-# captureExprBodyPattern = r"^return\s+" +
-#                          r"(" + anyPattern + r"?)" +
-#                          r";$"
-
-# capturePredicatePattern = r"^if\s+\((.*?)\)\s*\{" + blanksPattern +
-#                           r"return\s+true;?" + blanksPattern +
-#                           r"\}$"
-
-stars = r"\s*\n\s*\*\s?"
-
-atattrs = r"\s*\n\s*\*\s*@"
-
-
-def stripStars(text):
-    return re.sub(stars, '\n', text).strip()
-
-
-def parseTestEnvelope(src, name):
-    envelope = { 'testRecord': {} }
-    envelopeMatch = re.match(testEnvelopePattern, src)
-    if (envelopeMatch == None):
-        raise Exception('unrecognized: ' + name)
-    envelope['header'] = envelopeMatch.group(1).strip()
-    if (envelopeMatch.group(2)):
-        propTexts = re.split(atattrs, envelopeMatch.group(2))
-        envelope['commentary'] = stripStars(propTexts[0])
-        del propTexts[0]
-        for propText in propTexts:
-            # TODO: error check for mismatch
-            propName = re.match(r"^\w+", propText).group(0)
-            propVal = propText[len(propName):]
-
-            # Just till last one-time conversion
-            # strip optional initial colon or final semicolon.
-            # The initial colon is only stripped if it comes immediately
-            # after the identifier with no intervening whitespace.
-            propVal = re.sub(r"^:\s*", '', propVal, 1)
-            propVal = re.sub(r";\s*$", '', propVal, 1)
-            propVal = stripStars(propVal)
-            
-            if (propName in envelope['testRecord']):
-                raise Exception('duplicate: ' + propName)
-            envelope['testRecord'][propName] = propVal;
-    if (envelopeMatch.group(3)):
-        envelope['testRecord']['strict_only'] = '';
-    envelope['rest'] = envelopeMatch.group(4) # do not trim
-
-    # Just till last one time conversion
-    registerMatch = re.match(registerPattern, envelope['rest'])
-    if (registerMatch):
-        envelope['rest'] = registerMatch.group(1).strip()
-        envelope['registerExpr'] = registerMatch.group(2).strip()
-    else if ('ES5Harness.registerTest' in envelope['rest']):
-        raise Exception('Malformed harness? ' + name)
-    return envelope
-
-
-def functionSrcToProgramSrc(funcSrc):
-    cfnbMatch = re.match(captureFuncNameBodyPattern, funcSrc)
-    if (not cfnbMatch):
-        raise Exception('Could not recognize: "' + funcSrc + '"')
-    name = cfnbMatch.group(1).strip()
-    body = cfnbMatch.group(2).strip()
-
-    # Look for special cases
-
-    cebMatch = re.match(captureExprBodyPattern, body)
-    if (cebMatch):
-        return 'assertTruthy(' + cebMatch.group(1).strip() + ');'
-
-    cpMatch = re.match(capturePredicatePattern, body)
-    if (cpMatch):
-        return 'assertTruthy(' + cpMatch.group(1).strip() + ');'
-
-    # General case
-
-    return (funcSrc + '\n' +
-            'runTestCase(' + name + ');')
-
-
-def gatherOne(envelope, name):
-    # TODO(erights): implement by pattern match rather than evaluation
-    raise Exception('gatherOne not implemented yet')
-
-
-def transferProp(record, fromName, toName):
-    if    (((toName not in testRecord) or
-            (testRecord[toName] == '')) and
-           (fromName in testRecord)):
-        testRecord[toName] = testRecord[fromName]
-        del testRecord[fromName]
-
-
-# TODO: new midcap names
-# don't mask collisions -- give errors
-# if unrecognized names remain, give errors
-def normalizeProps(testRecord):
-    if    (('strict_only' not in testRecord) and 
-           ('strict' in testRecord) and
-           (testRecord['strict'] == 1)):
-        testRecord['strict_only'] = ''
-
-    if (testRecord['strict'] == 1):
-        del testRecord['strict']
-
-    if ('strict_mode_negative' in testRecord):
-        if ('strict_only' not in testRecord):
-            testRecord['strict_only'] = ''
-        transferProp(testRecord, 'strict_mode_negative', 'negative')
-
-     transferProp(testRecord, 'errortype', 'negative')
-     transferProp(testRecord, 'assertion', 'description')
-     transferProp(testRecord, 'assertion', 'commentary')
-
-
-def getGlobalScopeRecord(relPath):
-    # TODO(erights): implement
-    raise Exception('getGlobalScopeRecord not implemented yet')
-    
-    
-def parseTestRecord(inBase, relPath, name):
-    nextRelPath = relPath + [name]
-    nextPath = inBase + [name]
-    
-    
diff --git a/tools/packaging/parseTestRecord.py b/tools/packaging/parseTestRecord.py
new file mode 100644
index 0000000000..7a581a30a5
--- /dev/null
+++ b/tools/packaging/parseTestRecord.py
@@ -0,0 +1,65 @@
+#!/usr/bin/env python
+
+# Copyright 2011 by Google, Inc.  All rights reserved.
+# This code is governed by the BSD license found in the LICENSE file.
+
+# TODO: resolve differences with common.py and unify into one file.
+
+
+import logging
+import optparse
+import os
+from os import path
+import platform
+import re
+import subprocess
+import sys
+import tempfile
+import time
+
+# from TestCasePackagerConfig import *
+
+headerPatternStr = r"(?:(?:\s*\/\/.*)?\s*\n)*"
+captureCommentPatternStr = r"\/\*\*?((?:\s|\S)*?)\*\/\s*\n"
+anyPatternStr = r"(?:\s|\S)*"
+
+headerPattern = re.compile("^" + headerPatternStr)
+
+# Should match anything
+testRecordPattern = re.compile(r"^(" + headerPatternStr +
+                               r")(?:" + captureCommentPatternStr +
+                               r")?(" + anyPatternStr +
+                               r")$")
+
+stars = re.compile(r"\s*\n\s*\*\s?")
+atattrs = re.compile(r"\s*\n\s*\*\s*@")
+
+def stripStars(text):
+    return stars.sub('\n', text).strip()
+
+def stripHeader(src):
+    header = headerPattern.match(src).group(0)
+    return src[len(header):]
+
+def parseTestRecord(src, name):
+    testRecord = {}
+    match = testRecordPattern.match(src)
+    if match == None:
+        raise Exception('unrecognized: ' + name)
+    testRecord['header'] = match.group(1).strip()
+    testRecord['test'] = match.group(3) # do not trim
+    if match.group(2):
+        propTexts = atattrs.split(match.group(2))
+        testRecord['commentary'] = stripStars(propTexts[0])
+        del propTexts[0]
+        for propText in propTexts:
+            propMatch = re.match(r"^\w+", propText)
+            if propMatch == None:
+                raise Exception('Malformed "@" attribute: ' + name)
+            propName = propMatch.group(0)
+            propVal = stripStars(propText[len(propName):])
+            
+            if propName in testRecord:
+                raise Exception('duplicate: ' + propName)
+            testRecord[propName] = propVal;
+    return testRecord
diff --git a/tools/test262.py b/tools/packaging/test262.py
old mode 100644
new mode 100755
similarity index 62%
rename from tools/test262.py
rename to tools/packaging/test262.py
index c3a4792a0f..4ab3a3af19
--- a/tools/test262.py
+++ b/tools/packaging/test262.py
@@ -2,6 +2,11 @@
 # Copyright 2009 the Sputnik authors.  All rights reserved.
 # This code is governed by the BSD license found in the LICENSE file.
 
+# This is derived from sputnik.py, the Sputnik console test runner,
+# with elements from packager.py, which is separately
+# copyrighted. TODO: Refactor so there is less duplication between
+# test262.py and packager.py.
+
 
 import logging
 import optparse
@@ -13,25 +18,42 @@ import subprocess
 import sys
 import tempfile
 import time
+import xml.dom.minidom
+import datetime
+import shutil
+import json
+import stat
 
 
-class Test262Error(Exception):
+from parseTestRecord import parseTestRecord, stripHeader
+
+from packagerConfig import *
 
+class Test262Error(Exception):
   def __init__(self, message):
     self.message = message
 
-
 def ReportError(s):
   raise Test262Error(s)
 
 
+
+if not os.path.exists(EXCLUDED_FILENAME):
+    print "Cannot generate (JSON) test262 tests without a file," + \
+        " %s, showing which tests have been disabled!" % EXCLUDED_FILENAME
+    sys.exit(1)
+EXCLUDE_LIST = xml.dom.minidom.parse(EXCLUDED_FILENAME)
+EXCLUDE_LIST = EXCLUDE_LIST.getElementsByTagName("test")
+EXCLUDE_LIST = [x.getAttribute("id") for x in EXCLUDE_LIST]
+
+
 def BuildOptions():
   result = optparse.OptionParser()
   result.add_option("--command", default=None, help="The command-line to run")
   result.add_option("--tests", default=path.abspath('.'), 
                     help="Path to the tests")
   result.add_option("--cat", default=False, action="store_true",
-                    help="Print test source code")
+                    help="Print packaged test code that would be run")
   result.add_option("--summary", default=False, action="store_true",
                     help="Print summary after running tests")
   result.add_option("--full-summary", default=False, action="store_true",
@@ -40,7 +62,10 @@ def BuildOptions():
                     help="Test only strict mode")
   result.add_option("--non_strict_only", default=False, action="store_true", 
                     help="Test only non-strict mode")
-
+  # TODO: Once enough tests are made strict compat, change the default
+  # to "both"
+  result.add_option("--unmarked_default", default="non_strict", 
+                    help="default mode for tests of unspecified strictness")
   return result
 
 
@@ -51,16 +76,7 @@ def ValidateOptions(options):
     ReportError("Couldn't find test path '%s'" % options.tests)
 
 
-_PLACEHOLDER_PATTERN = re.compile(r"\{\{(\w+)\}\}")
-_INCLUDE_PATTERN = re.compile(r"\$INCLUDE\(\"(.*)\"\);")
-_SPECIAL_CALL_PATTERN = re.compile(r"\$([A-Z]+)(?=\()")
-
-
-_SPECIAL_CALLS = {
-  'ERROR': 'testFailed',
-  'FAIL': 'testFailed',
-  'PRINT': 'testPrint'
-}
+placeHolderPattern = re.compile(r"\{\{(\w+)\}\}")
 
 
 def IsWindows():
@@ -68,12 +84,6 @@ def IsWindows():
   return (p == 'Windows') or (p == 'Microsoft')
 
 
-def StripHeader(str):
-  while str.startswith('//') and "\n" in str:
-    str = str[str.index("\n")+1:]
-  return str.lstrip()
-
-
 class TempFile(object):
 
   def __init__(self, suffix="", prefix="tmp", text=False):
@@ -89,8 +99,7 @@ class TempFile(object):
     (self.fd, self.name) = tempfile.mkstemp(
         suffix = self.suffix,
         prefix = self.prefix,
-        text = self.text
-    )
+        text = self.text)
 
   def Write(self, str):
     os.write(self.fd, str)
@@ -127,7 +136,7 @@ class TestResult(object):
     mode = self.case.GetMode()
     if self.HasUnexpectedOutcome():
       if self.case.IsNegative():
-        print "%s was expected to fail in %s, but didn't" % (name, mode)
+        print "=== %s was expected to fail in %s, but didn't ===" % (name, mode)
       else:
         if long_format:
           print "=== %s failed in %s ===" % (name, mode)
@@ -164,11 +173,17 @@ class TestCase(object):
     self.suite = suite
     self.name = name
     self.full_path = full_path
-    self.contents = None
-    self.is_negative = None
     self.strict_mode = strict_mode
-    self.is_strict_only = None
-    self.is_non_strict_only = None
+    f = open(self.full_path)
+    self.contents = f.read()
+    f.close()
+    testRecord = parseTestRecord(self.contents, name)
+    self.test = testRecord["test"]
+    del testRecord["test"]
+    del testRecord["header"]
+    del testRecord["commentary"]
+    self.testRecord = testRecord;
+    
 
   def GetName(self):
     return path.join(*self.name)
@@ -182,60 +197,33 @@ class TestCase(object):
   def GetPath(self):
     return self.name
 
-  def GetRawContents(self):
-    if self.contents is None:
-      f = open(self.full_path)
-      self.contents = f.read()
-      f.close()
-    return self.contents
-
   def IsNegative(self):
-    if self.is_negative is None:
-      self.is_negative = ("@negative" in self.GetRawContents())
-    return self.is_negative
+    return 'negative' in self.testRecord
 
-  def IsStrictOnly(self):
-    if self.is_strict_only is None:
-      self.is_strict_only = ("@strict_only" in self.GetRawContents())
-    return self.is_strict_only
+  def IsOnlyStrict(self):
+    return 'onlyStrict' in self.testRecord
 
-  def IsNonStrictOnly(self):
-    if self.is_non_strict_only is None:
-      self.is_non_strict_only = ("@non_strict_only" in self.GetRawContents())
-    return self.is_non_strict_only
+  def IsNoStrict(self):
+    return 'noStrict' in self.testRecord
 
   def GetSource(self):
-    source = self.suite.GetInclude("framework.js", False) + \
-        self.suite.GetInclude("sta.js", False)
-    source += StripHeader(self.GetRawContents())
-    def IncludeFile(match):
-      return self.suite.GetInclude(match.group(1))
-    source = _INCLUDE_PATTERN.sub(IncludeFile, source)
-    def SpecialCall(match):
-      key = match.group(1)
-      return _SPECIAL_CALLS.get(key, match.group(0))
+    # "var testDescrip = " + str(self.testRecord) + ';\n\n' + \
+    source = self.suite.GetInclude("cth.js") + \
+        self.suite.GetInclude("sta.js") + \
+        self.suite.GetInclude("ed.js") + \
+        self.test + '\n'
+
     if self.strict_mode:
-      source = '"use strict";\nvar strict_mode = true;\n' + \
-          _SPECIAL_CALL_PATTERN.sub(SpecialCall, source)
+      source = '"use strict";\nvar strict_mode = true;\n' + source
     else:
-      source =  "var strict_mode = false; \n" + \
-          _SPECIAL_CALL_PATTERN.sub(SpecialCall, source)
+      source =  "var strict_mode = false; \n" + source
     return source
 
   def InstantiateTemplate(self, template, params):
     def GetParameter(match):
       key = match.group(1)
       return params.get(key, match.group(0))
-    return _PLACEHOLDER_PATTERN.sub(GetParameter, template)
-
-  def RunTestIn(self, command_template, tmp):
-    tmp.Write(self.GetSource())
-    tmp.Close()
-    command = self.InstantiateTemplate(command_template, {
-      'path': tmp.name
-    })
-    (code, out, err) = self.Execute(command)
-    return TestResult(code, out, err, self)
+    return placeHolderPattern.sub(GetParameter, template)
 
   def Execute(self, command):
     if IsWindows():
@@ -260,6 +248,15 @@ class TestCase(object):
       stderr.Dispose()
     return (code, out, err)
 
+  def RunTestIn(self, command_template, tmp):
+    tmp.Write(self.GetSource())
+    tmp.Close()
+    command = self.InstantiateTemplate(command_template, {
+      'path': tmp.name
+    })
+    (code, out, err) = self.Execute(command)
+    return TestResult(code, out, err, self)
+
   def Run(self, command_template):
     tmp = TempFile(suffix=".js", prefix="test262-", text=True)
     try:
@@ -298,11 +295,13 @@ def MakePlural(n):
 
 class TestSuite(object):
 
-  def __init__(self, root, strict_only, non_strict_only):
-    self.test_root = path.join(root, 'test', 'suite', 'converted')
+  def __init__(self, root, strict_only, non_strict_only, unmarked_default):
+    # TODO: derive from packagerConfig.py
+    self.test_root = path.join(root, 'test', 'suite')
     self.lib_root = path.join(root, 'test', 'harness')
     self.strict_only = strict_only
     self.non_strict_only = non_strict_only
+    self.unmarked_default = unmarked_default
     self.include_cache = { }
 
   def Validate(self):
@@ -325,41 +324,18 @@ class TestSuite(object):
         return True
     return False
 
-  def GetTimeZoneInfoInclude(self):
-    dst_attribs = GetDaylightSavingsAttribs()
-    if not dst_attribs:
-      return None
-    lines = []
-    for key in sorted(dst_attribs.keys()):
-      lines.append('var $DST_%s = %s;' % (key, str(dst_attribs[key])))
-    localtz = time.timezone / -3600
-    lines.append('var $LocalTZ = %i;' % localtz)
-    return "\n".join(lines)
-
-  def GetSpecialInclude(self, name):
-    if name == "environment.js":
-      return self.GetTimeZoneInfoInclude()
-    else:
-      return None
-
-  def GetInclude(self, name, strip_header=True):
-    key = (name, strip_header)
-    if not key in self.include_cache:
-      value = self.GetSpecialInclude(name)
-      if value:
-        self.include_cache[key] = value
+  def GetInclude(self, name):
+    if not name in self.include_cache:
+      static = path.join(self.lib_root, name)
+      if path.exists(static):
+        f = open(static)
+        contents = stripHeader(f.read())
+        contents = re.sub(r'\r\n', '\n', contents)
+        self.include_cache[name] = contents + "\n"
+        f.close()
       else:
-        static = path.join(self.lib_root, name)
-        if path.exists(static):
-          f = open(static)
-          contents = f.read()
-          if strip_header:
-            contents = StripHeader(contents)
-          self.include_cache[key] = contents + "\n"
-          f.close()
-        else:
-         self.include_cache[key] = ""
-    return self.include_cache[key]
+        ReportError("Can't find: " + static)
+    return self.include_cache[name]
 
   def EnumerateTests(self, tests):
     logging.info("Listing tests in %s", self.test_root)
@@ -379,14 +355,21 @@ class TestSuite(object):
           if self.ShouldRun(rel_path, tests):
             basename = path.basename(full_path)[:-3]
             name = rel_path.split(path.sep)[:-1] + [basename]
-            if not self.non_strict_only:
-              strict_case = TestCase(self, name, full_path, True)
-              if not strict_case.IsNonStrictOnly():
-                cases.append(strict_case)
-            if not self.strict_only:
-              non_strict_case = TestCase(self, name, full_path, False)
-              if not non_strict_case.IsStrictOnly():
-                cases.append(non_strict_case)
+            if EXCLUDE_LIST.count(basename) >= 1:
+              print 'Excluded: ' + basename
+            else:
+              if not self.non_strict_only:
+                strict_case = TestCase(self, name, full_path, True)
+                if not strict_case.IsNoStrict():
+                  if strict_case.IsOnlyStrict() or \
+                        self.unmarked_default in ['both', 'strict']:
+                    cases.append(strict_case)
+              if not self.strict_only:
+                non_strict_case = TestCase(self, name, full_path, False)
+                if not non_strict_case.IsOnlyStrict():
+                  if non_strict_case.IsNoStrict() or \
+                        self.unmarked_default in ['both', 'non_strict']:
+                    cases.append(non_strict_case)
     logging.info("Done listing tests")
     return cases
 
@@ -447,83 +430,14 @@ class TestSuite(object):
       cases[0].Print()
 
 
-def GetDaylightSavingsTimes():
-  # Is the given floating-point time in DST?
-  def IsDst(t):
-    return time.localtime(t)[-1]
-  # Binary search to find an interval between the two times no greater than
-  # delta where DST switches, returning the midpoint.
-  def FindBetween(start, end, delta):
-    while end - start > delta:
-      middle = (end + start) / 2
-      if IsDst(middle) == IsDst(start):
-        start = middle
-      else:
-        end = middle
-    return (start + end) / 2
-  now = time.time()
-  one_month = (30 * 24 * 60 * 60)
-  # First find a date with different daylight savings.  To avoid corner cases
-  # we try four months before and after today.
-  after = now + 4 * one_month
-  before = now - 4 * one_month
-  if IsDst(now) == IsDst(before) and IsDst(now) == IsDst(after):
-    logging.warning("Was unable to determine DST info.")
-    return None
-  # Determine when the change occurs between now and the date we just found
-  # in a different DST.
-  if IsDst(now) != IsDst(before):
-    first = FindBetween(before, now, 1)
-  else:
-    first = FindBetween(now, after, 1)
-  # Determine when the change occurs between three and nine months from the
-  # first.
-  second = FindBetween(first + 3 * one_month, first + 9 * one_month, 1)
-  # Find out which switch is into and which if out of DST
-  if IsDst(first - 1) and not IsDst(first + 1):
-    start = second
-    end = first
-  else:
-    start = first
-    end = second
-  return (start, end)
-
-
-def GetDaylightSavingsAttribs():
-  times = GetDaylightSavingsTimes()
-  if not times:
-    return None
-  (start, end) = times
-  def DstMonth(t):
-    return time.localtime(t)[1] - 1
-  def DstHour(t):
-    return time.localtime(t - 1)[3] + 1
-  def DstSunday(t):
-    if time.localtime(t)[2] > 15:
-      return "'last'"
-    else:
-      return "'first'"
-  def DstMinutes(t):
-    return (time.localtime(t - 1)[4] + 1) % 60
-  attribs = { }
-  attribs['start_month'] = DstMonth(start)
-  attribs['end_month'] = DstMonth(end)
-  attribs['start_sunday'] = DstSunday(start)
-  attribs['end_sunday'] = DstSunday(end)
-  attribs['start_hour'] = DstHour(start)
-  attribs['end_hour'] = DstHour(end)
-  attribs['start_minutes'] = DstMinutes(start)
-  attribs['end_minutes'] = DstMinutes(end)
-  return attribs
-
-
 def Main():
   parser = BuildOptions()
   (options, args) = parser.parse_args()
   ValidateOptions(options)
   test_suite = TestSuite(options.tests, 
                          options.strict_only, 
-                         options.non_strict_only)
+                         options.non_strict_only,
+                         options.unmarked_default)
   test_suite.Validate()
   if options.cat:
     test_suite.Print(args)
-- 
GitLab