java.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. import _common as common
  2. import ymake
  3. import json
  4. import os
  5. import base64
  6. import six
  7. DELIM = '================================'
  8. CONTRIB_JAVA_PREFIX = 'contrib/java/'
  9. def split_args(s): # TODO quotes, escapes
  10. return list(filter(None, s.split()))
  11. def extract_macro_calls(unit, macro_value_name, macro_calls_delim):
  12. if not unit.get(macro_value_name):
  13. return []
  14. return list(
  15. filter(
  16. None,
  17. map(split_args, unit.get(macro_value_name).replace('$' + macro_value_name, '').split(macro_calls_delim)),
  18. )
  19. )
  20. def extract_macro_calls2(unit, macro_value_name):
  21. if not unit.get(macro_value_name):
  22. return []
  23. calls = []
  24. for call_encoded_args in unit.get(macro_value_name).strip().split():
  25. call_args = json.loads(base64.b64decode(call_encoded_args))
  26. calls.append(call_args)
  27. return calls
  28. def onjava_module(unit, *args):
  29. args_delim = unit.get('ARGS_DELIM')
  30. if unit.get('YA_IDE_IDEA') != 'yes':
  31. return
  32. data = {
  33. 'BUNDLE_NAME': unit.name(),
  34. 'PATH': unit.path(),
  35. 'MODULE_TYPE': unit.get('MODULE_TYPE'),
  36. 'MODULE_ARGS': unit.get('MODULE_ARGS'),
  37. 'MANAGED_PEERS': '${MANAGED_PEERS}',
  38. 'MANAGED_PEERS_CLOSURE': '${MANAGED_PEERS_CLOSURE}',
  39. 'NON_NAMAGEABLE_PEERS': '${NON_NAMAGEABLE_PEERS}',
  40. 'EXCLUDE': extract_macro_calls(unit, 'EXCLUDE_VALUE', args_delim),
  41. 'JAVA_SRCS': extract_macro_calls(unit, 'JAVA_SRCS_VALUE', args_delim),
  42. 'JAVAC_FLAGS': extract_macro_calls(unit, 'JAVAC_FLAGS_VALUE', args_delim),
  43. 'ANNOTATION_PROCESSOR': extract_macro_calls(unit, 'ANNOTATION_PROCESSOR_VALUE', args_delim),
  44. 'JAR_INCLUDE_FILTER': extract_macro_calls(unit, 'JAR_INCLUDE_FILTER_VALUE', args_delim),
  45. 'JAR_EXCLUDE_FILTER': extract_macro_calls(unit, 'JAR_EXCLUDE_FILTER_VALUE', args_delim),
  46. # TODO remove when java test dart is in prod
  47. 'UNITTEST_DIR': unit.get('UNITTEST_DIR'),
  48. 'SYSTEM_PROPERTIES': extract_macro_calls(unit, 'SYSTEM_PROPERTIES_VALUE', args_delim),
  49. 'JVM_ARGS': extract_macro_calls(unit, 'JVM_ARGS_VALUE', args_delim),
  50. 'TEST_CWD': extract_macro_calls(unit, 'TEST_CWD_VALUE', args_delim),
  51. 'TEST_FORK_MODE': extract_macro_calls(unit, 'TEST_FORK_MODE', args_delim),
  52. 'SPLIT_FACTOR': extract_macro_calls(unit, 'TEST_SPLIT_FACTOR', args_delim),
  53. 'TIMEOUT': extract_macro_calls(unit, 'TEST_TIMEOUT', args_delim),
  54. 'TAG': extract_macro_calls(unit, 'TEST_TAGS_VALUE', args_delim),
  55. 'SIZE': extract_macro_calls(unit, 'TEST_SIZE_NAME', args_delim),
  56. 'DEPENDS': extract_macro_calls(unit, 'TEST_DEPENDS_VALUE', args_delim),
  57. 'IDEA_EXCLUDE': extract_macro_calls(unit, 'IDEA_EXCLUDE_DIRS_VALUE', args_delim),
  58. 'IDEA_RESOURCE': extract_macro_calls(unit, 'IDEA_RESOURCE_DIRS_VALUE', args_delim),
  59. 'IDEA_MODULE_NAME': extract_macro_calls(unit, 'IDEA_MODULE_NAME_VALUE', args_delim),
  60. 'FAKEID': extract_macro_calls(unit, 'FAKEID', args_delim),
  61. 'TEST_DATA': extract_macro_calls(unit, 'TEST_DATA_VALUE', args_delim),
  62. 'JAVA_FORBIDDEN_LIBRARIES': extract_macro_calls(unit, 'JAVA_FORBIDDEN_LIBRARIES_VALUE', args_delim),
  63. 'JDK_RESOURCE': 'JDK' + (unit.get('JDK_VERSION') or unit.get('JDK_REAL_VERSION') or '_DEFAULT'),
  64. }
  65. if unit.get('ENABLE_PREVIEW_VALUE') == 'yes' and (unit.get('JDK_VERSION') or unit.get('JDK_REAL_VERSION')) in (
  66. '17',
  67. '20',
  68. '21',
  69. '22',
  70. '23',
  71. ):
  72. data['ENABLE_PREVIEW'] = extract_macro_calls(unit, 'ENABLE_PREVIEW_VALUE', args_delim)
  73. if unit.get('SAVE_JAVAC_GENERATED_SRCS_DIR') and unit.get('SAVE_JAVAC_GENERATED_SRCS_TAR'):
  74. data['SAVE_JAVAC_GENERATED_SRCS_DIR'] = extract_macro_calls(unit, 'SAVE_JAVAC_GENERATED_SRCS_DIR', args_delim)
  75. data['SAVE_JAVAC_GENERATED_SRCS_TAR'] = extract_macro_calls(unit, 'SAVE_JAVAC_GENERATED_SRCS_TAR', args_delim)
  76. if unit.get('JAVA_ADD_DLLS_VALUE') == 'yes':
  77. data['ADD_DLLS_FROM_DEPENDS'] = extract_macro_calls(unit, 'JAVA_ADD_DLLS_VALUE', args_delim)
  78. if unit.get('ERROR_PRONE_VALUE') == 'yes':
  79. data['ERROR_PRONE'] = extract_macro_calls(unit, 'ERROR_PRONE_VALUE', args_delim)
  80. if unit.get('WITH_KOTLIN_VALUE') == 'yes':
  81. data['WITH_KOTLIN'] = extract_macro_calls(unit, 'WITH_KOTLIN_VALUE', args_delim)
  82. if unit.get('KOTLIN_JVM_TARGET'):
  83. data['KOTLIN_JVM_TARGET'] = extract_macro_calls(unit, 'KOTLIN_JVM_TARGET', args_delim)
  84. if unit.get('KOTLINC_FLAGS_VALUE'):
  85. data['KOTLINC_FLAGS'] = extract_macro_calls(unit, 'KOTLINC_FLAGS_VALUE', args_delim)
  86. if unit.get('KOTLINC_OPTS_VALUE'):
  87. data['KOTLINC_OPTS'] = extract_macro_calls(unit, 'KOTLINC_OPTS_VALUE', args_delim)
  88. if unit.get('DIRECT_DEPS_ONLY_VALUE') == 'yes':
  89. data['DIRECT_DEPS_ONLY'] = extract_macro_calls(unit, 'DIRECT_DEPS_ONLY_VALUE', args_delim)
  90. if unit.get('JAVA_EXTERNAL_DEPENDENCIES_VALUE'):
  91. valid = []
  92. for dep in sum(extract_macro_calls(unit, 'JAVA_EXTERNAL_DEPENDENCIES_VALUE', args_delim), []):
  93. if os.path.normpath(dep).startswith('..'):
  94. ymake.report_configure_error(
  95. '{}: {} - relative paths in JAVA_EXTERNAL_DEPENDENCIES is not allowed'.format(unit.path(), dep)
  96. )
  97. elif os.path.isabs(dep):
  98. ymake.report_configure_error(
  99. '{}: {} absolute paths in JAVA_EXTERNAL_DEPENDENCIES is not allowed'.format(unit.path(), dep)
  100. )
  101. else:
  102. valid.append(dep)
  103. if valid:
  104. data['EXTERNAL_DEPENDENCIES'] = [valid]
  105. if unit.get('MAKE_UBERJAR_VALUE') == 'yes':
  106. if unit.get('MODULE_TYPE') != 'JAVA_PROGRAM':
  107. ymake.report_configure_error('{}: UBERJAR supported only for JAVA_PROGRAM module type'.format(unit.path()))
  108. data['UBERJAR'] = extract_macro_calls(unit, 'MAKE_UBERJAR_VALUE', args_delim)
  109. data['UBERJAR_PREFIX'] = extract_macro_calls(unit, 'UBERJAR_PREFIX_VALUE', args_delim)
  110. data['UBERJAR_HIDE_EXCLUDE'] = extract_macro_calls(unit, 'UBERJAR_HIDE_EXCLUDE_VALUE', args_delim)
  111. data['UBERJAR_PATH_EXCLUDE'] = extract_macro_calls(unit, 'UBERJAR_PATH_EXCLUDE_VALUE', args_delim)
  112. data['UBERJAR_MANIFEST_TRANSFORMER_MAIN'] = extract_macro_calls(
  113. unit, 'UBERJAR_MANIFEST_TRANSFORMER_MAIN_VALUE', args_delim
  114. )
  115. data['UBERJAR_MANIFEST_TRANSFORMER_ATTRIBUTE'] = extract_macro_calls(
  116. unit, 'UBERJAR_MANIFEST_TRANSFORMER_ATTRIBUTE_VALUE', args_delim
  117. )
  118. data['UBERJAR_APPENDING_TRANSFORMER'] = extract_macro_calls(
  119. unit, 'UBERJAR_APPENDING_TRANSFORMER_VALUE', args_delim
  120. )
  121. data['UBERJAR_SERVICES_RESOURCE_TRANSFORMER'] = extract_macro_calls(
  122. unit, 'UBERJAR_SERVICES_RESOURCE_TRANSFORMER_VALUE', args_delim
  123. )
  124. if unit.get('WITH_JDK_VALUE') == 'yes':
  125. if unit.get('MODULE_TYPE') != 'JAVA_PROGRAM':
  126. ymake.report_configure_error(
  127. '{}: JDK export supported only for JAVA_PROGRAM module type'.format(unit.path())
  128. )
  129. data['WITH_JDK'] = extract_macro_calls(unit, 'WITH_JDK_VALUE', args_delim)
  130. # IMPORTANT before switching vcs_info.py to python3 the value was always evaluated to $YMAKE_PYTHON but no
  131. # code in java dart parser extracts its value only checks this key for existance.
  132. data['EMBED_VCS'] = [['yes']]
  133. # FORCE_VCS_INFO_UPDATE is responsible for setting special value of VCS_INFO_DISABLE_CACHE__NO_UID__
  134. macro_val = extract_macro_calls(unit, 'FORCE_VCS_INFO_UPDATE', args_delim)
  135. macro_str = macro_val[0][0] if macro_val and macro_val[0] and macro_val[0][0] else ''
  136. if macro_str and macro_str == 'yes':
  137. data['VCS_INFO_DISABLE_CACHE__NO_UID__'] = macro_val
  138. for java_srcs_args in data['JAVA_SRCS']:
  139. external = None
  140. for i in six.moves.range(len(java_srcs_args)):
  141. arg = java_srcs_args[i]
  142. if arg == 'EXTERNAL':
  143. if not i + 1 < len(java_srcs_args):
  144. continue # TODO configure error
  145. ex = java_srcs_args[i + 1]
  146. if ex in ('EXTERNAL', 'SRCDIR', 'PACKAGE_PREFIX', 'EXCLUDE'):
  147. continue # TODO configure error
  148. if external is not None:
  149. continue # TODO configure error
  150. external = ex
  151. if external:
  152. unit.onpeerdir(external)
  153. data = {k: v for k, v in six.iteritems(data) if v}
  154. dart = 'JAVA_DART: ' + six.ensure_str(base64.b64encode(six.ensure_binary(json.dumps(data)))) + '\n' + DELIM + '\n'
  155. unit.set_property(['JAVA_DART_DATA', dart])
  156. def on_add_java_style_checks(unit, *args):
  157. if unit.get('LINT_LEVEL_VALUE') != "none" and common.get_no_lint_value(unit) != 'none':
  158. unit.onadd_check(['JAVA_STYLE', unit.get('LINT_LEVEL_VALUE')] + list(args))
  159. def on_add_kotlin_style_checks(unit, *args):
  160. """
  161. ktlint can be disabled using NO_LINT() and NO_LINT(ktlint)
  162. """
  163. if unit.get('WITH_KOTLIN_VALUE') == 'yes':
  164. if common.get_no_lint_value(unit) == '':
  165. unit.onadd_check(['ktlint'] + list(args))
  166. def on_add_classpath_clash_check(unit, *args):
  167. jdeps_val = (unit.get('CHECK_JAVA_DEPS_VALUE') or '').lower()
  168. if jdeps_val and jdeps_val not in ('yes', 'no', 'strict'):
  169. ymake.report_configure_error('CHECK_JAVA_DEPS: "yes", "no" or "strict" required')
  170. if jdeps_val and jdeps_val != 'no':
  171. unit.onjava_test_deps(jdeps_val)
  172. def on_add_detekt_report_check(unit, *args):
  173. if unit.get('WITH_KOTLIN_VALUE') == 'yes' and unit.get('WITH_KOTLINC_PLUGIN_DETEKT') == 'yes':
  174. unit.onadd_check(['detekt.report'] + list(args))
  175. # Ymake java modules related macros
  176. def on_check_java_srcdir(unit, *args):
  177. args = list(args)
  178. if 'SKIP_CHECK_SRCDIR' in args:
  179. return
  180. for arg in args:
  181. if '$' not in arg:
  182. arc_srcdir = os.path.join(unit.get('MODDIR'), arg)
  183. abs_srcdir = unit.resolve(os.path.join("$S/", arc_srcdir))
  184. if not os.path.exists(abs_srcdir) or not os.path.isdir(abs_srcdir):
  185. unit.onsrcdir(os.path.join('${ARCADIA_ROOT}', arc_srcdir))
  186. return
  187. srcdir = common.resolve_common_const(unit.resolve_arc_path(arg))
  188. if srcdir and srcdir.startswith('$S'):
  189. abs_srcdir = unit.resolve(srcdir)
  190. if not os.path.exists(abs_srcdir) or not os.path.isdir(abs_srcdir):
  191. unit.onsrcdir(os.path.join('${ARCADIA_ROOT}', srcdir[3:]))
  192. def on_fill_jar_copy_resources_cmd(unit, *args):
  193. if len(args) == 4:
  194. varname, srcdir, base_classes_dir, reslist = tuple(args)
  195. package = ''
  196. else:
  197. varname, srcdir, base_classes_dir, package, reslist = tuple(args)
  198. dest_dir = os.path.join(base_classes_dir, *package.split('.')) if package else base_classes_dir
  199. var = unit.get(varname)
  200. var += ' && $FS_TOOLS copy_files {} {} {}'.format(
  201. srcdir if srcdir.startswith('"$') else '${CURDIR}/' + srcdir, dest_dir, reslist
  202. )
  203. unit.set([varname, var])
  204. def on_fill_jar_gen_srcs(unit, *args):
  205. varname, jar_type, srcdir, base_classes_dir, java_list, kt_list, res_list = tuple(args[0:7])
  206. resolved_srcdir = unit.resolve_arc_path(srcdir)
  207. if not resolved_srcdir.startswith('$') or resolved_srcdir.startswith('$S'):
  208. return
  209. if jar_type == 'SRC_JAR' and unit.get('SOURCES_JAR') != 'yes':
  210. return
  211. args_delim = unit.get('JAR_BUILD_SCRIPT_FLAGS_DELIM')
  212. exclude_pos = args.index('EXCLUDE')
  213. globs = ' '.join(args[7:exclude_pos])
  214. excludes = ' '.join(args[exclude_pos + 1 :])
  215. var = unit.get(varname)
  216. var += f' {args_delim} --append -d {srcdir} -s {java_list} -k {kt_list} -r {res_list} --include-patterns {globs}'
  217. if jar_type == 'SRC_JAR':
  218. var += ' --all-resources'
  219. if len(excludes) > 0:
  220. var += f' --exclude-patterns {excludes}'
  221. if unit.get('WITH_KOTLIN_VALUE') == 'yes':
  222. var += ' --resolve-kotlin'
  223. unit.set([varname, var])
  224. def on_check_run_java_prog_classpath(unit, *args):
  225. if len(args) != 1:
  226. ymake.report_configure_error(
  227. 'multiple CLASSPATH elements in RUN_JAVA_PROGRAM invocation no more supported. Use JAVA_RUNTIME_PEERDIR on the JAVA_PROGRAM module instead'
  228. )
  229. def extract_words(words, keys):
  230. kv = {}
  231. k = None
  232. for w in words:
  233. if w in keys:
  234. k = w
  235. else:
  236. if k not in kv:
  237. kv[k] = []
  238. kv[k].append(w)
  239. return kv
  240. def parse_words(words):
  241. kv = extract_words(words, {'OUT', 'TEMPLATE'})
  242. if 'TEMPLATE' not in kv:
  243. kv['TEMPLATE'] = ['template.tmpl']
  244. ws = []
  245. for item in ('OUT', 'TEMPLATE'):
  246. for i, word in list(enumerate(kv[item])):
  247. if word == 'CUSTOM_PROPERTY':
  248. ws += kv[item][i:]
  249. kv[item] = kv[item][:i]
  250. templates = kv['TEMPLATE']
  251. outputs = kv['OUT']
  252. if len(outputs) < len(templates):
  253. ymake.report_configure_error('To many arguments for TEMPLATE parameter')
  254. return
  255. if ws and ws[0] != 'CUSTOM_PROPERTY':
  256. ymake.report_configure_error('''Can't parse {}'''.format(ws))
  257. custom_props = []
  258. for item in ws:
  259. if item == 'CUSTOM_PROPERTY':
  260. custom_props.append([])
  261. else:
  262. custom_props[-1].append(item)
  263. props = []
  264. for p in custom_props:
  265. if not p:
  266. ymake.report_configure_error('Empty CUSTOM_PROPERTY')
  267. continue
  268. props.append('-B')
  269. if len(p) > 1:
  270. props.append(six.ensure_str(base64.b64encode(six.ensure_binary("{}={}".format(p[0], ' '.join(p[1:]))))))
  271. else:
  272. ymake.report_configure_error('CUSTOM_PROPERTY "{}" value is not specified'.format(p[0]))
  273. for i, o in enumerate(outputs):
  274. yield o, templates[min(i, len(templates) - 1)], props
  275. def ongenerate_script(unit, *args):
  276. for out, tmpl, props in parse_words(list(args)):
  277. unit.on_add_gen_java_script([out, tmpl] + list(props))
  278. def on_jdk_version_macro_check(unit, *args):
  279. if len(args) != 1:
  280. unit.message(["error", "Invalid syntax. Single argument required."])
  281. jdk_version = args[0]
  282. available_versions = (
  283. '11',
  284. '17',
  285. '20',
  286. '21',
  287. '22',
  288. '23',
  289. )
  290. if jdk_version not in available_versions:
  291. ymake.report_configure_error(
  292. "Invalid jdk version: {}. {} are available".format(jdk_version, available_versions)
  293. )
  294. if int(jdk_version) >= 19 and unit.get('WITH_JDK_VALUE') != 'yes' and unit.get('MODULE_TAG') == 'JAR_RUNNABLE':
  295. msg = (
  296. "Missing WITH_JDK() macro for JDK version >= 19"
  297. # temporary link with additional explanation
  298. ". For more info see https://clubs.at.yandex-team.ru/arcadia/28543"
  299. )
  300. ymake.report_configure_error(msg)
  301. def _maven_coords_for_project(unit, project_dir):
  302. parts = project_dir.split('/')
  303. g = '.'.join(parts[2:-2])
  304. a = parts[-2]
  305. v = parts[-1]
  306. c = ''
  307. pom_path = unit.resolve(os.path.join('$S', project_dir, 'pom.xml'))
  308. if os.path.exists(pom_path):
  309. import xml.etree.ElementTree as et
  310. try:
  311. with open(pom_path, 'rb') as f:
  312. root = et.fromstring(f.read())
  313. for xpath in ('./{http://maven.apache.org/POM/4.0.0}artifactId', './artifactId'):
  314. artifact = root.find(xpath)
  315. if artifact is not None:
  316. artifact = artifact.text
  317. if a != artifact and a.startswith(artifact):
  318. c = a[len(artifact) :].lstrip('-_')
  319. a = artifact
  320. break
  321. except Exception as e:
  322. raise Exception(f"Can't parse {pom_path}: {str(e)}") from None
  323. return '{}:{}:{}:{}'.format(g, a, v, c)
  324. def on_setup_maven_export_coords_if_need(unit, *args):
  325. if not unit.enabled('MAVEN_EXPORT'):
  326. return
  327. unit.set(['MAVEN_EXPORT_COORDS_GLOBAL', _maven_coords_for_project(unit, args[0])])
  328. def _get_classpath(unit, dir):
  329. if dir.startswith(CONTRIB_JAVA_PREFIX):
  330. return '\\"{}\\"'.format(_maven_coords_for_project(unit, dir).rstrip(':'))
  331. else:
  332. return 'project(\\":{}\\")'.format(dir.replace('/', ':'))
  333. def on_setup_project_coords_if_needed(unit, *args):
  334. if not unit.enabled('EXPORT_GRADLE'):
  335. return
  336. project_dir = args[0]
  337. if project_dir.startswith(CONTRIB_JAVA_PREFIX):
  338. value = '{}'.format(_get_classpath(unit, project_dir).rstrip(':'))
  339. else:
  340. value = 'project(\\":{}\\")'.format(project_dir.replace('/', ':'))
  341. unit.set(['EXPORT_GRADLE_CLASSPATH', value])