blob: e9dbd35e5d80fad9f7de061275c779134d4b6294 [file] [log] [blame] [edit]
#!/usr/bin/env vpython3
# Copyright 2023 The Chromium Authors
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Given a .build_config.json file, generates an Eclipse JDT project that can
be used with the "Language Support for Java™ by Red Hat" Visual Studio Code
extension. See //docs/vscode.md for details.
"""
import argparse
import logging
import json
import os
import sys
import xml.etree.ElementTree
sys.path.append(os.path.join(os.path.dirname(__file__), 'gyp'))
from util import build_utils
sys.path.append(os.path.join(os.path.dirname(__file__), os.pardir))
import gn_helpers
def _WithoutSuffix(string, suffix):
if not string.endswith(suffix):
raise ValueError(f'{string!r} does not end with {suffix!r}')
return string[:-len(suffix)]
def _GetJavaRoot(path):
# The authoritative way to determine the Java root for a given source file is
# to parse the source code and extract the package and class names, but let's
# keep things simple and use some heuristics to try to guess the Java root
# from the file path instead.
while True:
dirname, basename = os.path.split(path)
if not basename:
raise RuntimeError(f'Unable to determine the Java root for {path!r}')
if basename in ('java', 'src'):
return path
if basename in ('javax', 'org', 'com'):
return dirname
path = dirname
def _ProcessSourceFile(output_dir, source_file_path, source_dirs):
source_file_path = os.path.normpath(os.path.join(output_dir,
source_file_path))
java_root = _GetJavaRoot(source_file_path)
logging.debug('Extracted java root `%s` from source file path `%s`',
java_root, source_file_path)
source_dirs.add(java_root)
def _ProcessSourcesFile(output_dir, sources_file_path, source_dirs):
for source_file_path in build_utils.ReadSourcesList(
os.path.join(output_dir, sources_file_path)):
_ProcessSourceFile(output_dir, source_file_path, source_dirs)
def _ProcessBuildConfigFile(output_dir, build_config_path, source_dirs, libs,
already_processed_build_config_files,
android_sdk_build_tools_version):
if build_config_path in already_processed_build_config_files:
return
already_processed_build_config_files.add(build_config_path)
logging.info('Processing build config: %s', build_config_path)
with open(os.path.join(output_dir, build_config_path)) as build_config_file:
build_config = json.load(build_config_file)
deps_info = build_config['deps_info']
target_sources_file = deps_info.get('target_sources_file')
if target_sources_file is not None:
_ProcessSourcesFile(output_dir, target_sources_file, source_dirs)
else:
unprocessed_jar_path = deps_info.get('unprocessed_jar_path')
if unprocessed_jar_path is not None:
lib_path = os.path.normpath(os.path.join(output_dir,
unprocessed_jar_path))
logging.debug('Found lib `%s', lib_path)
libs.add(lib_path)
input_srcjars = os.path.join(output_dir,
_WithoutSuffix(build_config_path, '.build_config.json'),
'generated_java', 'input_srcjars')
if os.path.exists(input_srcjars):
source_dirs.add(input_srcjars)
android = build_config.get('android')
if android is not None:
# This works around an issue where the language server complains about
# `java.lang.invoke.LambdaMetafactory` not being found. The normal Android
# build process is fine with this class being missing because d8 removes
# references to LambdaMetafactory from the bytecode - see:
# https://jakewharton.com/androids-java-8-support/#native-lambdas
# When JDT builds the code, d8 doesn't run, so the references are still
# there. Fortunately, the Android SDK provides a convenience JAR to fill
# that gap in:
# //third_party/android_sdk/public/build-tools/*/core-lambda-stubs.jar
libs.add(
os.path.normpath(
os.path.join(
output_dir,
os.path.dirname(build_config['android']['sdk_jars'][0]),
os.pardir, os.pardir, 'build-tools',
android_sdk_build_tools_version, 'core-lambda-stubs.jar')))
for dep_config in deps_info['deps_configs']:
_ProcessBuildConfigFile(output_dir, dep_config, source_dirs, libs,
already_processed_build_config_files,
android_sdk_build_tools_version)
def _GenerateClasspathEntry(kind, path):
classpathentry = xml.etree.ElementTree.Element('classpathentry')
classpathentry.set('kind', kind)
classpathentry.set('path', path)
return classpathentry
def _GenerateProject(source_dirs, libs, output_dir):
classpath = xml.etree.ElementTree.Element('classpath')
for source_dir in sorted(source_dirs):
classpath.append(_GenerateClasspathEntry('src', source_dir))
for lib in sorted(libs):
classpath.append(_GenerateClasspathEntry('lib', lib))
classpath.append(
_GenerateClasspathEntry('output', os.path.join(output_dir, 'jdt_output')))
xml.etree.ElementTree.ElementTree(classpath).write(
'.classpath', encoding='unicode')
print('Generated .classpath', file=sys.stderr)
with open('.project', 'w') as f:
f.write("""<?xml version="1.0" encoding="UTF-8"?>
<projectDescription>
<name>chromium</name>
<buildSpec>
<buildCommand>
<name>org.eclipse.jdt.core.javabuilder</name>
<arguments />
</buildCommand>
</buildSpec>
<natures><nature>org.eclipse.jdt.core.javanature</nature></natures>
</projectDescription>
""")
print('Generated .project', file=sys.stderr)
# Tell the Eclipse compiler not to use java.lang.invoke.StringConcatFactory
# in the generated bytecodes as the class is unavailable in Android.
os.makedirs('.settings', exist_ok=True)
with open('.settings/org.eclipse.jdt.core.prefs', 'w') as f:
f.write("""eclipse.preferences.version=1
org.eclipse.jdt.core.compiler.codegen.useStringConcatFactory=disabled
""")
print('Generated .settings', file=sys.stderr)
def _ParseArguments(argv):
parser = argparse.ArgumentParser(
description=
'Given Chromium Java build config files, generates an Eclipse JDT '
'project that can be used with the "Language Support for Java™ by '
'Red Hat" Visual Studio Code extension. See //docs/vscode.md '
'for details.')
parser.add_argument(
'--output-dir',
required=True,
help='Relative path to the output directory, e.g. "out/Debug"')
parser.add_argument(
'--build-config',
action='append',
required=True,
help='Path to the .build_config.json file to use as input, relative to '
'`--output-dir`. May be repeated.')
return parser.parse_args(argv)
def main(argv):
build_utils.InitLogging('GENERATE_VSCODE_CLASSPATH_DEBUG')
assert os.path.exists('.gn'), 'This script must be run from the src directory'
args = _ParseArguments(argv)
output_dir = args.output_dir
build_vars = gn_helpers.ReadBuildVars(output_dir)
source_dirs = set()
libs = set()
already_processed_build_config_files = set()
for build_config_path in args.build_config:
_ProcessBuildConfigFile(output_dir, build_config_path, source_dirs, libs,
already_processed_build_config_files,
build_vars['android_sdk_build_tools_version'])
logging.info('Done processing %d build config files',
len(already_processed_build_config_files))
_GenerateProject(source_dirs, libs, output_dir)
if __name__ == '__main__':
sys.exit(main(sys.argv[1:]))