blob: b5a3ffa15d7ba06e42f4b7914ef0951dabf8868c [file] [log] [blame]
Primiano Tucci3faad742018-05-16 19:30:48 +01001# Copyright (C) 2018 The Android Open Source Project
2#
3# Licensed under the Apache License, Version 2.0 (the "License");
4# you may not use this file except in compliance with the License.
5# You may obtain a copy of the License at
6#
7# http://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS,
11# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12# See the License for the specific language governing permissions and
13# limitations under the License.
14
15import("../gn/perfetto.gni")
16import("../gn/wasm.gni")
17import("../protos/perfetto/trace_processor/proto_files.gni")
18
19ui_dist_dir = "$root_build_dir/ui"
20nodejs_root = "../buildtools/nodejs"
21nodejs_bin = rebase_path("$nodejs_root/bin", root_build_dir)
22
23# +----------------------------------------------------------------------------+
24# | The outer "ui" target to just ninja -C out/xxx ui |
25# +----------------------------------------------------------------------------+
26group("ui") {
27 deps = [
28 ":bundle",
29 ":index",
30 ":wasm",
31 ]
32}
33
34# +----------------------------------------------------------------------------+
35# | Template used to run node binaries using the hermetic node toolchain. |
36# +----------------------------------------------------------------------------+
37template("node_bin") {
38 action(target_name) {
39 forward_variables_from(invoker,
40 [
41 "inputs",
42 "outputs",
43 ])
44 deps = [
45 ":node_modules",
46 ]
47 if (defined(invoker.deps)) {
48 deps += invoker.deps
49 }
50 script = "../gn/standalone/build_tool_wrapper.py"
51 _node_cmd = invoker.node_cmd
52 args = [
53 "--path=$nodejs_bin",
54 "node",
55 rebase_path("node_modules/.bin/$_node_cmd", root_build_dir),
56 ] + invoker.args
57 }
58}
59
60# +----------------------------------------------------------------------------+
61# | Bundles all *.js files together resolving CommonJS require() deps. |
62# +----------------------------------------------------------------------------+
63
64# Bundle together all js sources into a bundle.js file, that will ultimately be
65# included by the .html files.
66node_bin("bundle") {
67 deps = [
68 ":transpile_all_ts",
69 ]
70 node_cmd = "browserify"
71 inputs = [
72 "$ui_dist_dir/main.js",
73 ]
74 outputs = [
75 "$ui_dist_dir/bundle.js",
76 ]
77 args = [
78 rebase_path(inputs[0], root_build_dir),
79 "-d",
80 "-o",
81 rebase_path(outputs[0], root_build_dir),
82 ]
83}
84
85# +----------------------------------------------------------------------------+
86# | Protobuf: gen rules to create .js and .d.ts files from protos. |
87# +----------------------------------------------------------------------------+
88proto_gen_dir = "$ui_dist_dir/gen"
89
90node_bin("protos_to_js") {
91 inputs = []
92 foreach(proto, trace_processor_protos) {
93 inputs += [ "../protos/perfetto/trace_processor/$proto.proto" ]
94 }
95 outputs = [
96 "$proto_gen_dir/protos.js",
97 ]
98 node_cmd = "pbjs"
99 args = [
100 "-t",
101 "static-module",
102 "-w",
103 "commonjs",
104 "-p",
105 rebase_path("../protos", root_build_dir),
106 "-o",
107 rebase_path(outputs[0], root_build_dir),
108 ] + rebase_path(inputs, root_build_dir)
109}
110
111# Protobuf.js requires to first generate .js files from the .proto and then
112# create .ts definitions for them.
113node_bin("protos_to_ts") {
114 deps = [
115 ":protos_to_js",
116 ]
117 inputs = [
118 "$proto_gen_dir/protos.js",
119 ]
120 outputs = [
121 "$proto_gen_dir/protos.d.ts",
122 ]
123 node_cmd = "pbts"
124 args = [
125 "-p",
126 rebase_path("../protos", root_build_dir),
127 "-o",
128 rebase_path(outputs[0], root_build_dir),
129 rebase_path(inputs[0], root_build_dir),
130 ]
131}
132
133# +----------------------------------------------------------------------------+
134# | TypeScript: transpiles all *.ts into .js |
135# +----------------------------------------------------------------------------+
136
137# Builds all .ts sources in the repo that are reachable from |sources|.
138node_bin("transpile_all_ts") {
139 sources = [
140 "main.ts",
141 ]
142 deps = [
143 ":dist_symlink",
144 ":protos_to_ts",
145 ]
146 inputs = sources + [ "tsconfig.json" ]
147 outputs = [
148 "$ui_dist_dir/main.js",
149 ]
150
151 # Find all *.ts files and pass them as input triggers. This does NOT affect
152 # which files will get built. This defines only the set of files that, when
153 # changes, will re-trigger this rule in ninja. The files that get transpiled
154 # are the transitive dependencies of |sources|.
155 all_ts_files = exec_script("../gn/standalone/glob.py",
156 [
157 "--root=" + rebase_path(".", root_build_dir),
158 "--filter=*.ts",
159 "--exclude=node_modules",
160 "--exclude=dist",
161 ],
162 "list lines",
163 [ "." ])
164 inputs += all_ts_files
165
166 node_cmd = "tsc"
167 args = [
168 "--project",
169 rebase_path(".", root_build_dir),
170 "--outDir",
171 rebase_path(ui_dist_dir, root_build_dir),
172 ]
173}
174
175# +----------------------------------------------------------------------------+
176# | Copy rules: copy assets into the dist directory. |
177# +----------------------------------------------------------------------------+
178copy("index") {
179 sources = [
180 "index.html",
181 ]
182 outputs = [
183 "$ui_dist_dir/index.html",
184 ]
185}
186
187copy("wasm") {
188 deps = [
189 "//src/trace_processor:trace_processor.js($wasm_toolchain)",
190 "//src/trace_processor:trace_processor.wasm($wasm_toolchain)",
191 ]
192 sources = [
193 "$root_build_dir/wasm/trace_processor.js",
194 "$root_build_dir/wasm/trace_processor.wasm",
195 ]
196 outputs = [
197 "$ui_dist_dir/wasm/{{source_file_part}}",
198 ]
199}
200
201# +----------------------------------------------------------------------------+
202# | Node JS: run npm install and creates a symlink in the out directory. |
203# +----------------------------------------------------------------------------+
204
205action("check_node_exists") {
206 script = "../gn/standalone/check_buildtool_exists.py"
207 args = [ nodejs_bin ]
208 inputs = []
209 outputs = [
210 "$target_out_dir/ignored",
211 ]
212}
213
214# Creates a symlink from out/xxx/ui/node_modules -> ../../../ui/node_modules.
215# This allows to run browserify and other node tools from the out/xxx directory.
216action("node_modules_symlink") {
217 deps = [
218 ":check_node_exists",
219 ]
220 inputs = [
221 "node_modules",
222 ]
223 outputs = [
224 "$ui_dist_dir/node_modules",
225 ]
226 script = "../gn/standalone/build_tool_wrapper.py"
227 args = [
228 "/bin/ln",
229 "-fns",
230 rebase_path(inputs[0], ui_dist_dir),
231 rebase_path(outputs[0], root_build_dir),
232 ]
233}
234
235# Runs npm install.
236action("node_modules") {
237 script = "../gn/standalone/build_tool_wrapper.py"
238 stamp_file = "$ui_dist_dir/.$target_name.stamp"
239 args = [
240 "--chdir",
241 rebase_path(".", root_build_dir),
242 "--stamp",
243 rebase_path(stamp_file, root_build_dir),
244 ]
245 args += [
246 "--path=$nodejs_bin",
247 "node",
248 rebase_path("$nodejs_root/bin/npm", "."),
249 "install",
250 "--no-save",
251 "--silent",
252 ]
253 inputs = [
254 "package.json",
255 "package-lock.json",
256 ]
257 outputs = [
258 stamp_file,
259 ]
260 deps = [
261 ":node_modules_symlink",
262 ]
263}
264
265# Creates a symlink from //ui/dist -> ../../out/xxx/ui. Used only for
266# autocompletion in IDEs. The problem this is solving is that in tsconfig.json
267# we can't possibly know the path to ../../out/xxx for outDir. Instead, we set
268# outDir to "./dist" and create a symlink on the first build.
269action("dist_symlink") {
270 script = "../gn/standalone/build_tool_wrapper.py"
271 stamp_file = "$ui_dist_dir/.$target_name.stamp"
272 args = [
273 "--stamp",
274 rebase_path(stamp_file, root_build_dir),
275 "/bin/ln",
276 "-fns",
277 rebase_path(ui_dist_dir, "."),
278 rebase_path("dist", root_build_dir),
279 ]
280 inputs = [
281 "$root_build_dir",
282 ]
283 outputs = [
284 stamp_file,
285 ]
286}