Skip to content

Commit

Permalink
Add build rules to build kotlin library and binary (#25222)
Browse files Browse the repository at this point in the history
  • Loading branch information
yufengwangca authored and pull[bot] committed Jun 26, 2023
1 parent 22ccc44 commit 912bdcb
Show file tree
Hide file tree
Showing 2 changed files with 461 additions and 0 deletions.
146 changes: 146 additions & 0 deletions build/chip/java/kotlinc_runner.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,146 @@
#!/usr/bin/env python
# Copyright (c) 2023 Project CHIP Authors
#
# 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.

# Copyright 2015 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Wrapper script to run kotlinc command as an action with gn."""

import argparse
import json
import os
import subprocess
import sys

EXIT_SUCCESS = 0
EXIT_FAILURE = 1


def IsExecutable(path):
"""Returns whether file at |path| exists and is executable.
Args:
path: absolute or relative path to test.
Returns:
True if the file at |path| exists, False otherwise.
"""
return os.path.isfile(path) and os.access(path, os.X_OK)


def FindCommand(command):
"""Looks up for |command| in PATH.
Args:
command: name of the command to lookup, if command is a relative or absolute
path (i.e. contains some path separator) then only that path will be
tested.
Returns:
Full path to command or None if the command was not found.
On Windows, this respects the PATHEXT environment variable when the
command name does not have an extension.
"""
fpath, _ = os.path.split(command)
if fpath:
if IsExecutable(command):
return command

if sys.platform == 'win32':
# On Windows, if the command does not have an extension, cmd.exe will
# try all extensions from PATHEXT when resolving the full path.
command, ext = os.path.splitext(command)
if not ext:
exts = os.environ['PATHEXT'].split(os.path.pathsep)
else:
exts = [ext]
else:
exts = ['']

for path in os.environ['PATH'].split(os.path.pathsep):
for ext in exts:
path = os.path.join(path, command) + ext
if IsExecutable(path):
return path

return None


def ReadBuildConfig(build_config):
with open(build_config, 'r') as file:
return json.load(file)


def ComputeClasspath(build_config_json):
unique_jars = build_config_json['deps_info']['deps_jars']
if sys.platform == 'win32':
return ";".join(unique_jars)
else:
return ":".join(unique_jars)


def main():
kotlin_path = FindCommand('kotlinc')
if not kotlin_path:
sys.stderr.write('kotlinc: command not found\n')
sys.exit(EXIT_FAILURE)

parser = argparse.ArgumentParser('Kotkinc runner')
parser.add_argument(
'--classdir',
dest='classdir',
required=True,
help='Directory that will contain class files')
parser.add_argument(
'--outfile',
dest='outfile',
required=True,
help='Output file containing a list of classes')
parser.add_argument(
'--build-config',
dest='build_config',
required=True,
help='Build config')
parser.add_argument(
'rest', metavar='KOTLINC_ARGS', nargs='*', help='Argumets to pass to kotlinc')

args = parser.parse_args()
if not os.path.isdir(args.classdir):
os.makedirs(args.classdir, exist_ok=True)

build_config_json = ReadBuildConfig(args.build_config)
classpath = ComputeClasspath(build_config_json)
kotlin_args = [kotlin_path]
if classpath:
kotlin_args += ["-classpath", classpath]

retcode = subprocess.check_call(kotlin_args + args.rest)
if retcode != EXIT_SUCCESS:
return retcode

with open(args.outfile, 'wt') as f:
prefixlen = len(args.classdir) + 1
for root, dirnames, filenames in os.walk(args.classdir):
for filename in filenames:
if filename.endswith('.class'):
f.write(os.path.join(root[prefixlen:], filename))
f.write('\n')

return EXIT_SUCCESS


if __name__ == '__main__':
sys.exit(main())
Loading

0 comments on commit 912bdcb

Please sign in to comment.