#!/usr/bin/env python

# Copyright (c) 2012 Google Inc. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.

"""
Verifies that device and simulator bundles are built correctly.
"""

import plistlib
import TestGyp
import os
import struct
import subprocess
import sys
import tempfile
import TestMac

def CheckFileType(file, expected):
  proc = subprocess.Popen(['lipo', '-info', file], stdout=subprocess.PIPE)
  o = proc.communicate()[0].strip()
  assert not proc.returncode
  if not expected in o:
    print 'File: Expected %s, got %s' % (expected, o)
    test.fail_test()

def HasCerts():
  # Because the bots do not have certs, don't check them if there are no
  # certs available.
  proc = subprocess.Popen(['security','find-identity','-p', 'codesigning',
                           '-v'], stdout=subprocess.PIPE)
  return "0 valid identities found" not in proc.communicate()[0].strip()

def CheckSignature(file):
  proc = subprocess.Popen(['codesign', '-v', file], stdout=subprocess.PIPE)
  o = proc.communicate()[0].strip()
  assert not proc.returncode
  if "code object is not signed at all" in o:
    print 'File %s not properly signed.' % (file)
    test.fail_test()

def CheckEntitlements(file, expected_entitlements):
  with tempfile.NamedTemporaryFile() as temp:
    proc = subprocess.Popen(['codesign', '--display', '--entitlements',
                             temp.name, file], stdout=subprocess.PIPE)
    o = proc.communicate()[0].strip()
    assert not proc.returncode
    data = temp.read()
  entitlements = ParseEntitlements(data)
  if not entitlements:
    print 'No valid entitlements found in %s.' % (file)
    test.fail_test()
  if entitlements != expected_entitlements:
    print 'Unexpected entitlements found in %s.' % (file)
    test.fail_test()

def ParseEntitlements(data):
  if len(data) < 8:
    return None
  magic, length = struct.unpack('>II', data[:8])
  if magic != 0xfade7171 or length != len(data):
    return None
  return data[8:]

def GetXcodeVersionValue(type):
  args = ['xcodebuild', '-version', '-sdk', 'iphoneos', type]
  job = subprocess.Popen(args, stdout=subprocess.PIPE)
  return job.communicate()[0].strip()

def GetMachineBuild():
  args = ['sw_vers', '-buildVersion']
  job = subprocess.Popen(args, stdout=subprocess.PIPE)
  return job.communicate()[0].strip()

def CheckPlistvalue(plist, key, expected):
  if key not in plist:
    print '%s not set in plist' % key
    test.fail_test()
    return
  actual = plist[key]
  if actual != expected:
    print 'File: Expected %s, got %s for %s' % (expected, actual, key)
    test.fail_test()

def CheckPlistNotSet(plist, key):
  if key in plist:
    print '%s should not be set in plist' % key
    test.fail_test()
    return

def ConvertBinaryPlistToXML(path):
  proc = subprocess.call(['plutil', '-convert', 'xml1', path],
                         stdout=subprocess.PIPE)

if sys.platform == 'darwin':
  test = TestGyp.TestGyp(formats=['ninja', 'xcode'])

  test.run_gyp('test-device.gyp', chdir='app-bundle')

  test_configs = ['Default-iphoneos', 'Default']
  for configuration in test_configs:
    test.set_configuration(configuration)
    test.build('test-device.gyp', 'test_app', chdir='app-bundle')
    result_file = test.built_file_path('Test App Gyp.app/Test App Gyp',
                                       chdir='app-bundle')
    test.must_exist(result_file)
    info_plist = test.built_file_path('Test App Gyp.app/Info.plist',
                                      chdir='app-bundle')
    plist = plistlib.readPlist(info_plist)
    xcode_version = TestMac.Xcode.Version()
    if xcode_version >= '0720':
      if len(plist) != 23:
        print 'plist should have 23 entries, but it has %s' % len(plist)
        test.fail_test()

    # Values that will hopefully never change.
    CheckPlistvalue(plist, 'CFBundleDevelopmentRegion', 'English')
    CheckPlistvalue(plist, 'CFBundleExecutable', 'Test App Gyp')
    CheckPlistvalue(plist, 'CFBundleIdentifier', 'com.google.Test App Gyp')
    CheckPlistvalue(plist, 'CFBundleInfoDictionaryVersion', '6.0')
    CheckPlistvalue(plist, 'CFBundleName', 'Test App Gyp')
    CheckPlistvalue(plist, 'CFBundlePackageType', 'APPL')
    CheckPlistvalue(plist, 'CFBundleShortVersionString', '1.0')
    CheckPlistvalue(plist, 'CFBundleSignature', 'ause')
    CheckPlistvalue(plist, 'CFBundleVersion', '1')
    CheckPlistvalue(plist, 'NSMainNibFile', 'MainMenu')
    CheckPlistvalue(plist, 'NSPrincipalClass', 'NSApplication')
    CheckPlistvalue(plist, 'UIDeviceFamily', [1, 2])

    # Values that get pulled from xcodebuild.
    machine_build = GetMachineBuild()
    platform_version = GetXcodeVersionValue('ProductVersion')
    sdk_build = GetXcodeVersionValue('ProductBuildVersion')
    xcode_build = TestMac.Xcode.Build()

    # Xcode keeps changing what gets included in executable plists, and it
    # changes between device and simuator builds.  Allow the strictest tests for
    # Xcode 7.2 and above.
    if xcode_version >= '0720':
      CheckPlistvalue(plist, 'BuildMachineOSBuild', machine_build)
      CheckPlistvalue(plist, 'DTCompiler', 'com.apple.compilers.llvm.clang.1_0')
      CheckPlistvalue(plist, 'DTPlatformVersion', platform_version)
      CheckPlistvalue(plist, 'DTSDKBuild', sdk_build)
      CheckPlistvalue(plist, 'DTXcode', xcode_version)
      CheckPlistvalue(plist, 'DTXcodeBuild', xcode_build)
      CheckPlistvalue(plist, 'MinimumOSVersion', '8.0')


    if configuration == 'Default-iphoneos':
      platform_name = 'iphoneos'
      CheckFileType(result_file, 'armv7')
      CheckPlistvalue(plist, 'CFBundleSupportedPlatforms', ['iPhoneOS'])
      # Apple keeps changing their mind.
      if xcode_version >= '0720':
        CheckPlistvalue(plist, 'DTPlatformBuild', sdk_build)
    else:
      platform_name = 'iphonesimulator'
      CheckFileType(result_file, 'i386')
      CheckPlistvalue(plist, 'CFBundleSupportedPlatforms', ['iPhoneSimulator'])
      if xcode_version >= '0720':
        CheckPlistvalue(plist, 'DTPlatformBuild', '')

    CheckPlistvalue(plist, 'DTPlatformName', platform_name)
    CheckPlistvalue(plist, 'DTSDKName', platform_name + platform_version)


    if HasCerts() and configuration == 'Default-iphoneos':
      test.build('test-device.gyp', 'sig_test', chdir='app-bundle')
      result_file = test.built_file_path('sigtest.app/sigtest',
                                         chdir='app-bundle')
      CheckSignature(result_file)
      info_plist = test.built_file_path('sigtest.app/Info.plist',
                                        chdir='app-bundle')

      plist = plistlib.readPlist(info_plist)
      CheckPlistvalue(plist, 'UIDeviceFamily', [1])

      entitlements_file = test.built_file_path('sig_test.xcent',
                                               chdir='app-bundle')
      if os.path.isfile(entitlements_file):
        expected_entitlements = open(entitlements_file).read()
        CheckEntitlements(result_file, expected_entitlements)

  test.pass_test()