/
symbol_graph.rb
109 lines (94 loc) · 3.7 KB
/
symbol_graph.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
# frozen_string_literal: true
require 'set'
require 'jazzy/symbol_graph/graph'
require 'jazzy/symbol_graph/constraint'
require 'jazzy/symbol_graph/symbol'
require 'jazzy/symbol_graph/relationship'
require 'jazzy/symbol_graph/sym_node'
require 'jazzy/symbol_graph/ext_node'
require 'jazzy/symbol_graph/ext_key'
# This is the top-level symbolgraph driver that deals with
# figuring out arguments, running the tool, and loading the
# results.
module Jazzy
module SymbolGraph
# Find swift symbol graph files, either having been passed
# in directly, or generated by running`swift symbolgraph-extract`
# with configured args.
# Then parse the results, and return as JSON in SourceKit[ten]
# format.
def self.build(module_config)
if module_config.symbolgraph_directory.nil?
Dir.mktmpdir do |tmp_dir|
args = arguments(module_config, tmp_dir)
Executable.execute_command('swift',
args.unshift('symbolgraph-extract'),
true) # raise on error
parse_symbols(tmp_dir)
end
else
parse_symbols(module_config.symbolgraph_directory.to_s)
end
end
# Figure out the args to pass to symbolgraph-extract
def self.arguments(module_config, output_path)
if module_config.module_name.empty?
raise 'error: `--swift-build-tool symbolgraph` requires `--module`.'
end
user_args = module_config.build_tool_arguments.join
if user_args =~ /-(?:module-name|minimum-access-level|output-dir)/
raise 'error: `--build-tool-arguments` for ' \
"`--swift-build-tool symbolgraph` can't use `-module`, " \
'`-minimum-access-level`, or `-output-dir`.'
end
# Default set
args = [
'-module-name', module_config.module_name,
'-minimum-access-level', 'private',
'-output-dir', output_path,
'-skip-synthesized-members'
]
# Things user can override
args += ['-sdk', sdk(module_config)] unless user_args =~ /-sdk/
args += ['-target', target] unless user_args =~ /-target/
args += ['-F', module_config.source_directory.to_s] unless user_args =~ /-F(?!s)/
args += ['-I', module_config.source_directory.to_s] unless user_args =~ /-I/
args + module_config.build_tool_arguments
end
# Parse the symbol files in the given directory
def self.parse_symbols(directory)
Dir[directory + '/*.symbols.json'].sort.map do |filename|
# The @ part is for extensions in our module (before the @)
# of types in another module (after the @).
File.basename(filename) =~ /(.*?)(@(.*?))?\.symbols/
module_name = Regexp.last_match[1]
ext_module_name = Regexp.last_match[3] || module_name
json = File.read(filename)
{
filename =>
Graph.new(json, module_name, ext_module_name).to_sourcekit,
}
end.to_json
end
# Get the SDK path. On !darwin this just isn't needed.
def self.sdk(module_config)
`xcrun --show-sdk-path --sdk #{module_config.sdk}`.chomp
end
# Guess a default LLVM target. Feels like the tool should figure this
# out from sdk + the binary somehow?
def self.target
`swift -version` =~ /Target: (.*?)$/
Regexp.last_match[1] || 'x86_64-apple-macosx10.15'
end
# This is a last-ditch fallback for when symbolgraph doesn't
# provide a name - at least conforming external types to local
# protocols.
def self.demangle(usr)
args = %w[demangle -simplified -compact].append(usr.sub(/^s:/, 's'))
output, = Executable.execute_command('swift', args, true)
output.chomp
rescue StandardError
usr
end
end
end