# Copyright 2014 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import os
import os.path
import tempfile
import subprocess
import time
import sys
import textwrap
import its.device
def main():
"""Run all the automated tests, saving intermediate files, and producing
a summary/report of the results.
Script should be run from the top-level CameraITS directory.
"""
SKIP_RET_CODE = 101
# Not yet mandated tests
NOT_YET_MANDATED = {
"scene0":[
"test_jitter"
],
"scene1":[
"test_ae_precapture_trigger",
"test_crop_region_raw",
"test_ev_compensation_advanced",
"test_ev_compensation_basic",
"test_yuv_plus_jpeg"
],
"scene2":[],
"scene3":[],
"scene4":[],
"scene5":[],
"sensor_fusion":[]
}
# Get all the scene0 and scene1 tests, which can be run using the same
# physical setup.
scenes = ["scene0", "scene1", "scene2", "scene3", "scene4", "scene5"]
scene_req = {
"scene0" : None,
"scene1" : "A grey card covering at least the middle 30% of the scene",
"scene2" : "A picture containing human faces",
"scene3" : "A chart containing sharp edges like ISO 12233",
"scene4" : "A specific test page of a circle covering at least the "
"middle 50% of the scene. See CameraITS.pdf section 2.3.4 "
"for more details",
"scene5" : "Capture images with a diffuser attached to the camera. See "
"CameraITS.pdf section 2.3.4 for more details",
"sensor_fusion" : "Rotating checkboard pattern. See "
"sensor_fusion/SensorFusion.pdf for detailed "
"instructions. Note that this test will be skipped "
"on devices not supporting REALTIME camera timestamp."
"If that is the case, no scene setup is required and "
"you can just answer Y when being asked if the scene "
"is okay"
}
scene_extra_args = {
"scene5" : ["doAF=False"]
}
tests = []
for d in scenes:
tests += [(d,s[:-3],os.path.join("tests", d, s))
for s in os.listdir(os.path.join("tests",d))
if s[-3:] == ".py"]
tests.sort()
# Make output directories to hold the generated files.
topdir = tempfile.mkdtemp()
print "Saving output files to:", topdir, "\n"
device_id = its.device.get_device_id()
device_id_arg = "device=" + device_id
print "Testing device " + device_id
camera_ids = []
for s in sys.argv[1:]:
if s[:7] == "camera=" and len(s) > 7:
camera_ids.append(s[7:])
# user doesn't specify camera id, run through all cameras
if not camera_ids:
camera_ids_path = os.path.join(topdir, "camera_ids.txt")
out_arg = "out=" + camera_ids_path
cmd = ['python',
os.path.join(os.getcwd(),"tools/get_camera_ids.py"), out_arg,
device_id_arg]
retcode = subprocess.call(cmd,cwd=topdir)
assert(retcode == 0)
with open(camera_ids_path, "r") as f:
for line in f:
camera_ids.append(line.replace('\n', ''))
print "Running ITS on the following cameras:", camera_ids
for camera_id in camera_ids:
# Loop capturing images until user confirm test scene is correct
camera_id_arg = "camera=" + camera_id
print "Preparing to run ITS on camera", camera_id
os.mkdir(os.path.join(topdir, camera_id))
for d in scenes:
os.mkdir(os.path.join(topdir, camera_id, d))
print "Start running ITS on camera: ", camera_id
# Run each test, capturing stdout and stderr.
summary = "ITS test result summary for camera " + camera_id + "\n"
numpass = 0
numskip = 0
numnotmandatedfail = 0
numfail = 0
prev_scene = ""
for (scene,testname,testpath) in tests:
if scene != prev_scene and scene_req[scene] != None:
out_path = os.path.join(topdir, camera_id, scene+".jpg")
out_arg = "out=" + out_path
scene_arg = "scene=" + scene_req[scene]
extra_args = scene_extra_args.get(scene, [])
cmd = ['python',
os.path.join(os.getcwd(),"tools/validate_scene.py"),
camera_id_arg, out_arg, scene_arg, device_id_arg] + \
extra_args
retcode = subprocess.call(cmd,cwd=topdir)
assert(retcode == 0)
print "Start running tests for", scene
prev_scene = scene
cmd = ['python', os.path.join(os.getcwd(),testpath)] + \
sys.argv[1:] + [camera_id_arg]
outdir = os.path.join(topdir,camera_id,scene)
outpath = os.path.join(outdir,testname+"_stdout.txt")
errpath = os.path.join(outdir,testname+"_stderr.txt")
t0 = time.time()
with open(outpath,"w") as fout, open(errpath,"w") as ferr:
retcode = subprocess.call(cmd,stderr=ferr,stdout=fout,cwd=outdir)
t1 = time.time()
if retcode == 0:
retstr = "PASS "
numpass += 1
elif retcode == SKIP_RET_CODE:
retstr = "SKIP "
numskip += 1
elif retcode != 0 and testname in NOT_YET_MANDATED[scene]:
retstr = "FAIL*"
numnotmandatedfail += 1
else:
retstr = "FAIL "
numfail += 1
msg = "%s %s/%s [%.1fs]" % (retstr, scene, testname, t1-t0)
print msg
summary += msg + "\n"
if retcode != 0 and retcode != SKIP_RET_CODE:
# Dump the stderr if the test fails
with open (errpath, "r") as error_file:
errors = error_file.read()
summary += errors + "\n"
if numskip > 0:
skipstr = ", %d test%s skipped" % (numskip, "s" if numskip > 1 else "")
else:
skipstr = ""
test_result = "\n%d / %d tests passed (%.1f%%)%s" % (
numpass + numnotmandatedfail, len(tests) - numskip,
100.0 * float(numpass + numnotmandatedfail) / (len(tests) - numskip)
if len(tests) != numskip else 100.0,
skipstr)
print test_result
summary += test_result + "\n"
if numnotmandatedfail > 0:
msg = "(*) tests are not yet mandated"
print msg
summary += msg + "\n"
result = numfail == 0
print "Reporting ITS result to CtsVerifier"
summary_path = os.path.join(topdir, camera_id, "summary.txt")
with open(summary_path, "w") as f:
f.write(summary)
its.device.report_result(device_id, camera_id, result, summary_path)
print "ITS tests finished. Please go back to CtsVerifier and proceed"
if __name__ == '__main__':
main()