Skip to content

Commit b012253

Browse files
authored
Merge pull request #164 from mathworks/autotrace
enable specifying folders in AutoTrace options, closes #163
2 parents 5b3cff0 + 961c824 commit b012253

File tree

12 files changed

+266
-46
lines changed

12 files changed

+266
-46
lines changed

auto-instrumentation/+opentelemetry/+autoinstrument/AutoTrace.m

+38-15
Original file line numberDiff line numberDiff line change
@@ -58,9 +58,15 @@
5858
options.AdditionalFiles {mustBeText}
5959
options.AutoDetectFiles (1,1) {mustBeNumericOrLogical} = true
6060
end
61+
% check for anonymous function
62+
fs = functions(startfun);
63+
if fs.type == "anonymous"
64+
error("opentelemetry:autoinstrument:AutoTrace:AnonymousFunction", ...
65+
"Anonymous functions are not supported.");
66+
end
6167
obj.StartFunction = startfun;
6268
startfunname = func2str(startfun);
63-
processFileInput(startfunname); % validate startfun
69+
startfunname = processFileInput(startfunname); % validate startfun
6470
if options.AutoDetectFiles
6571
if isdeployed
6672
% matlab.codetools.requiredFilesAndProducts is not
@@ -76,15 +82,16 @@
7682
end
7783
else
7884
% only include the input file, not its dependencies
79-
files = string(which(startfunname));
85+
files = startfunname;
8086
end
8187
% add extra files, this is intended for files
8288
% matlab.codetools.requiredFilesAndProducts somehow missed
8389
if isfield(options, "AdditionalFiles")
84-
incfiles = string(options.AdditionalFiles);
85-
for i = 1:numel(incfiles)
86-
incfiles(i) = which(incfiles(i)); % get the full path
87-
processFileInput(incfiles(i)); % validate additional file
90+
incinput = string(options.AdditionalFiles);
91+
incfiles = [];
92+
for i = 1:numel(incinput)
93+
% validate additional file
94+
incfiles = [incfiles; processFileOrFolderInput(incinput(i))]; %#ok<AGROW>
8895
end
8996
files = union(files, incfiles);
9097
end
@@ -94,9 +101,11 @@
94101

95102
% filter out excluded files
96103
if isfield(options, "ExcludeFiles")
97-
excfiles = string(options.ExcludeFiles);
98-
for i = 1:numel(excfiles)
99-
excfiles(i) = which(excfiles(i)); % get the full path
104+
excinput = string(options.ExcludeFiles);
105+
excfiles = [];
106+
for i = 1:numel(excinput)
107+
% validate exclude file
108+
excfiles = [excfiles; processFileOrFolderInput(excinput(i))]; %#ok<AGROW>
100109
end
101110
files = setdiff(files, excfiles);
102111
end
@@ -155,15 +164,13 @@ function handleError(obj, ME)
155164
end
156165

157166
% check input file is valid
158-
function processFileInput(f)
167+
function f = processFileInput(f)
159168
f = string(f); % force into a string
160-
if startsWith(f, '@') % check for anonymous function
161-
error("opentelemetry:autoinstrument:AutoTrace:AnonymousFunction", ...
162-
replace(f, "\", "\\") + " is an anonymous function and is not supported.");
163-
end
164169
[~,~,fext] = fileparts(f); % check file extension
165170
filetype = exist(f, "file"); % check file type
166-
if ~(filetype == 2 && ismember(fext, ["" ".m" ".mlx"]))
171+
if filetype == 2 && ismember(fext, ["" ".m" ".mlx"])
172+
f = string(which(f));
173+
else
167174
if exist(f, "builtin")
168175
error("opentelemetry:autoinstrument:AutoTrace:BuiltinFunction", ...
169176
replace(f, "\", "\\") + " is a builtin function and is not supported.");
@@ -172,4 +179,20 @@ function processFileInput(f)
172179
replace(f, "\", "\\") + " is not found or is not a valid MATLAB file with a .m or .mlx extension.");
173180
end
174181
end
182+
end
183+
184+
% check input file or folder is valid
185+
function f = processFileOrFolderInput(f)
186+
f = string(f); % force into a string
187+
if isfolder(f)
188+
% expand the directory
189+
mfileinfo = dir(fullfile(f, "*.m"));
190+
mfiles = fullfile(string({mfileinfo.folder}), string({mfileinfo.name}));
191+
mlxfileinfo = dir(fullfile(f, "*.mlx"));
192+
mlxfiles = fullfile(string({mlxfileinfo.folder}), string({mlxfileinfo.name}));
193+
f = [mfiles; mlxfiles];
194+
else
195+
% file
196+
f = processFileInput(f);
197+
end
175198
end

test/autotrace_examples/example1/best_fit_line.m test/autotrace_examples/linearfit_example/best_fit_line.m

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
function yf = best_fit_line(x, y)
2-
% example code for testing auto instrumentation
2+
% Fit a straight line on input data
33

44
% Copyright 2024 The MathWorks, Inc.
55

test/autotrace_examples/example1/generate_data.m test/autotrace_examples/linearfit_example/generate_data.m

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,11 @@
11
function [x, y] = generate_data(n)
2-
% example code for testing auto instrumentation
2+
% Generate random data with n data points
33

44
% Copyright 2024 The MathWorks, Inc.
55

66
% check input is valid
77
if ~(isnumeric(n) && isscalar(n))
8-
error("autotrace_examples:example1:generate_data:InvalidN", ...
8+
error("autotrace_examples:linearfit_example:generate_data:InvalidN", ...
99
"Input must be a numeric scalar");
1010
end
1111

test/autotrace_examples/example1/example1.m test/autotrace_examples/linearfit_example/linearfit_example.m

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
1-
function yf = example1(n)
2-
% example code for testing auto instrumentation. Input n is the number of
1+
function yf = linearfit_example(n)
2+
% Example code for testing auto instrumentation. Input n is the number of
33
% data points.
44

55
% Copyright 2024 The MathWorks, Inc.

test/autotrace_examples/example1/example1_trycatch.m test/autotrace_examples/linearfit_example/linearfit_example_trycatch.m

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
1-
function yf = example1_trycatch(at, n)
2-
% example code for testing auto instrumentation. This example should not
1+
function yf = linearfit_example_trycatch(at, n)
2+
% Example code for testing auto instrumentation. This example should not
33
% use beginTrace method and instead should be called directly.
44

55
% Copyright 2024 The MathWorks, Inc.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
function yf = best_fit_line(x, y)
2+
% Fit a straight line on input data and manually start and end two spans.
3+
4+
% Copyright 2024 The MathWorks, Inc.
5+
6+
tr = opentelemetry.trace.getTracer("ManualInstrument");
7+
8+
sp1 = startSpan(tr, "polyfit");
9+
coefs = polyfit(x, y, 1);
10+
endSpan(sp1);
11+
12+
sp2 = startSpan(tr, "polyval");
13+
yf = polyval(coefs , x);
14+
endSpan(sp2);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
function [x, y] = generate_data(n)
2+
% Generate random data with n data points and manually start and end a span.
3+
4+
% Copyright 2024 The MathWorks, Inc.
5+
6+
% check input is valid
7+
if ~(isnumeric(n) && isscalar(n))
8+
error("autotrace_examples:linearfit_example:generate_data:InvalidN", ...
9+
"Input must be a numeric scalar");
10+
end
11+
12+
% generate some random data
13+
a = 1.5;
14+
b = 0.8;
15+
sigma = 5;
16+
x = 1:n;
17+
18+
% start a span
19+
tr = opentelemetry.trace.getTracer("ManualInstrument");
20+
sp = startSpan(tr, "compute_y");
21+
y = a * x + b + sigma * randn(1, n);
22+
endSpan(sp);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
function yf = manual_instrumented_example(n)
2+
% Example code for testing auto and manual instrumentation together.
3+
% Input n is the number of data points.
4+
5+
% Copyright 2024 The MathWorks, Inc.
6+
7+
[x, y] = generate_data(n);
8+
yf = best_fit_line(x,y);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
function x = subfolder_helper1(x)
2+
% example code for testing auto instrumentation, helper function
3+
4+
% Copyright 2024 The MathWorks, Inc.
5+
6+
x = x * 2;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
function x = subfolder_helper2(x)
2+
% example code for testing auto instrumentation, helper function
3+
4+
% Copyright 2024 The MathWorks, Inc.
5+
6+
x = x * 3;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
function x = subfolder_example
2+
% Example code for testing auto instrumentation, with some helper functions
3+
% in a subfolder
4+
5+
% Copyright 2024 The MathWorks, Inc.
6+
7+
x = 10;
8+
x = subfolder_helper1(x);
9+
x = subfolder_helper2(x);
10+

0 commit comments

Comments
 (0)