Scripts¶
Working with Scripts¶
Fityk can run two kinds of scripts:
Fityk scripts composed of the commands described in previous sections,
and Lua scripts (extension
.lua
), in the Lua language.
Scripts are executed using the exec command:
exec file1.fit
exec file2.lua
exec file3.fit.gz # read script compressed with gzip
Note
Fityk can save its state to a script (info state > file.fit
).
It can also save all commands executed (directly or via GUI) in the session
to a script (info history > file.fit
).
Embedded Lua interpreter can execute any program in Lua 5.2.
One-liners can be run with command lua
:
=-> lua print(_VERSION)
Lua 5.1
=-> lua print(os.date("Today is %A."))
Today is Thursday.
=-> lua for n,f in F:all_functions() do print(n, f, f:get_template_name()) end
0 %_1 Constant
1 %_2 Cycle
(The Lua print
function in fityk is redefined to show the output
in the GUI instead of writing to stdout
).
Like in the Lua interpreter, =
at the beginning of line can be used
to save some typing:
=-> = os.date("Today is %A.")
Today is Thursday.
Similarly, exec=
also interprets the rest of line
as Lua expressions, but this time the resulting string is executed
as a fityk command:
=-> = string.format("fit @%d", math.random(0,5))
fit @4
=-> exec= string.format("fit @%d", math.random(0,5))
# fitting random dataset (useless example)
The embedded Lua interpreter interacts with the rest of the program
through the global object F
:
=-> = F:get_info("version")
Fityk 1.2.1
All methods of F
are documented in the section Fityk library API.
A few more examples.
Let’s say that we work with a number of datasets, and for each of them
we want to save output of the info peaks
command to a file
named original-data-filename.out. This can be done in one line:
=-> @*: lua F:execute("info peaks >'%s.out'" % F:get_info("filename"))
Now a more complex script that would need to be put into a file
(with extension .lua
) and run with exec
.
:
-- load data from files file01.dat, file02.dat, ... file13.dat
for i=1,13 do
F:execute("@+ < file%02d.dat:0:1::" % i)
end
-- print some statistics about the loaded data
n = F:get_dataset_count()
print(n .. " datasets loaded.")
total_max_y = 0
for i=0, n-1 do
max_y = F:calculate_expr("max(y)", i)
if max_y > total_max_y then
total_max_y = max_y
end
end
print("The largest y: " .. total_max_y)
If a fityk command executed from Lua script fails, the whole script is stopped, unless you catch the error:
-- wrap F:execute() in pcall to handle possible errors
status, err = pcall(function() F:execute("fit") end)
if status == false then
print("Error: " .. err)
end
The Lua interpreter was added in ver. 1.0.3. If you need help with writing Lua scripts - feel free to ask. Usage scenarios give us a better idea what functions should be available from the Lua interface.
Fityk also has a simple mechanism to interact with external programs.
It is useful mostly on Unix systems. !
runs a program,
exec!
runs a program, reads its standard output and executes it
as a Fityk script.
Here is an example of using Unix utilties echo
, ls
and head
to load the newest CIF file from the current directory:
=-> ! pwd
/home/wojdyr/fityk/data
=-> ! ls -t *.cif | head -1
lab6_3-2610-q.cif
=-> exec! echo "@+ < $(ls -t *.cif | head -1)"
> @+ < lab6_3-2610-q.cif
2300 points. No explicit std. dev. Set as sqrt(y)
Fityk DSL¶
As was described in Command Line, each line has a syntax:
[[@…:] [with …] command [“;” command]…] [#comment]
The datasets listed before the colon (:
) make a foreach loop.
Here is a silly example:
=-> $a=0
=-> @0 @0 @0: $a={$a+1}; print $a
1
2
3
Command that follows the colon is run for each specified dataset in the context of that dataset. This is to say that:
=-> @2 @4: guess Voigt
is equivalent to:
=-> use @2
=-> guess Voigt
=-> use @4
=-> guess Voigt
(except that the latter sets permenently default dataset to @4
.
@*
stands for all datasets, from @0
to the last one.
Usually, when working with multiple datasets, one executes a command either for a single dataset or for all of them:
=-> @3: guess Voigt # just for @3
=-> @*: guess Voigt # for all datasets
The whole line is parsed and partly validated before the execution. This may lead to unexpected errors when the line has multiple semicolon-separated commands:
=-> $a=4; print $a # print gives unexpected error
Error: undefined variable: $a
=-> $b=2
=-> $b=4; print $b # $b is already defined at the check time
4
Therefore, it is recommended to have one command in one line.
Grammar¶
The grammar is expressed in EBNF-like notation:
(*this is a comment*)
A*
means 0 or more occurrences of A.A+
means 1 or more occurrences of A.A % B
meansA (B A)*
and the%
operator has the highest precedence. For example:term % "+" comment
is the same asterm ("+" term)* comment
.The colon
:
in quoted string means that the string can be shortened, e.g."del:ete"
means that any ofdel
,dele
,delet
anddelete
can be used.
The functions that can be used in p_expr
and v_expr
are available
here and here, respectively.
v_expr
contains only a subset of functions from p_expr
(partly,
because we need to calculate symbolical derivatives of v_expr
)
Line structure
line ::= [statement
] [comment
] statement ::= [Dataset+ ":"] [with_opts
]command
% ";" with_opts ::= "w:ith" (Lname "="value
) % "," comment ::= "#" AllChars*
Commands
The kCmd* names in the comments correspond to constants in the code.
command ::= ( "deb:ug" RestOfLine | (*kCmdDebug*) "def:ine"define
| (*kCmdDefine*) "del:ete"delete
| (*kCmdDelete*) "del:ete"delete_points
| (*kCmdDeleteP*) "e:xecute"exec
| (*kCmdExec*) "f:it"fit
| (*kCmdFit*) "g:uess"guess
| (*kCmdGuess*) "i:nfo"info_arg
% "," [redir
] | (*kCmdInfo*) "l:ua" RestOfLine | (*kCmdLua*) "=" RestOfLine | (*kCmdLua*) "pl:ot" [range
] [range
] Dataset* [redir
] | (*kCmdPlot*) "p:rint"print_args
[redir
] | (*kCmdPrint*) "quit" | (*kCmdQuit*) "reset" | (*kCmdReset*) "s:et" (Lname "="value
) % "," | (*kCmdSet*) "sleep"expr
| (*kCmdSleep*) "title" "="filename
| (*kCmdTitle*) "undef:ine" Uname % "," | (*kCmdUndef*) "use" Dataset | (*kCmdUse*) "!" RestOfLine | (*kCmdShell*) Dataset "<"load_arg
| (*kCmdLoad*) Dataset "="dataset_expr
| (*kCmdDatasetTr*) Funcname "="func_rhs
| (*kCmdNameFunc*)param_lhs
"="v_expr
| (*kCmdAssignParam*) Varname "="v_expr
| (*kCmdNameVar*) Varname "=" "copy" "("var_id
")" | (*kCmdNameVar*)model_id
("="|"+=")model_rhs
| (*kCmdChangeModel*) (p_attr
"["expr
"]" "="p_expr
) % "," | (*kCmdPointTr*) (p_attr
"="p_expr
) % "," | (*kCmdAllPointsTr*) "M" "="expr
) (*kCmdResizeP*)
Other rules
define ::= Uname "(" (Lname [ "="v_expr
]) % "," ")" "=" (v_expr
|component_func
% "+" | "x" "<"v_expr
"?"component_func
":"component_func
) component_func ::= Uname "("v_expr
% "," ")" delete ::= (Varname |func_id
| Dataset | "file"filename
) % "," delete_points ::= "("p_expr
")" exec ::=filename
| "!" RestOfLine | "=" RestOfLine fit ::= [Number] [Dataset*] | "undo" | "redo" | "history" Number | "clear_history" guess ::= [Funcname "="] Uname ["(" (Lname "="v_expr
) % "," ")"] [range
] info_arg ::= ...TODO print_args ::= [("all" | ("if"p_expr
":")] (p_expr
| QuotedString | "title" | "filename") % "," redir ::= (">"|">>")filename
value ::= (Lname | QuotedString |expr
) (*value type depends on the option*) model_rhs ::= "0" |func_id
|func_rhs
|model_id
| "copy" "("model_id
")" func_rhs ::= Uname "(" ([Lname "="]v_expr
) % "," ")" | "copy" "("func_id
")" load_arg ::=filename
Lname* | "." p_attr ::= ("X" | "Y" | "S" | "A") model_id ::= [Dataset "."] ("F"|"Z") func_id ::= Funcname |model_id
"[" Number "]" param_lhs ::= Funcname "." Lname |model_id
"[" (Number | "*") "]" "." Lname var_id ::= Varname |func_id
"." Lname range ::= "[" [expr
] ":" [expr
] "]" filename ::= QuotedString | NonblankString
Mathematical expressions
expr ::= expr_or ? expr_or : expr_or
expr_or ::= expr_and % "or"
expr_and ::= expr_not % "and"
expr_not ::= "not" expr_not | comparison
comparison ::= arith % ("<"|">"|"=="|">="|"<="|"!=")
arith ::= term % ("+"|"-")
term ::= factor % ("*"|"/")
factor ::= ('+'|'-') factor | power
power ::= atom ['**' factor]
atom ::= Number | "true" | "false" | "pi" |
math_func | braced_expr | ?others?
math_func ::= "sqrt" "(" expr ")" |
"gamma" "(" expr ")" |
...
braced_expr ::= "{" [Dataset+ ":"] p_expr
"}"
The atom
rule also accepts some fityk expressions, such as $variable,
%function.parameter, %function(expr), etc.
p_expr
and v_expr
are similar to expr
,
but they use additional variables in the atom
rule.
p_expr
recognizes n
, M
, x
, y
, s
, a
, X
, Y
,
S
and A
. All of them but n
and M
can be indexed
(e.g. x[4]
). Example: (x+x[n-1])/2
.
v_expr
uses all unknown names (Lname
) as variables
(example: a+b*x^2
).
Only a subset of functions (math_func
) from expr
is supported.
The tilde (~
) can be used to create simple-variables (~5
),
optionally with a domain in square brackets (~5[1:6]
).
Since v_expr
is used to define variables and user-defined functions,
the program calculates symbolically derivatives of v_expr
.
That is why not all the function from expr
are supported
(they may be added in the future).
dataset_expr
supports very limited set of operators and a few functions
that take Dataset token as argument (example: @0 - shirley_bg(@0)
).
Lexer
Below, some of the tokens produced by the fityk lexer are defined.
The lexer is context-dependend: NonblankString
and RestOfLine
are produced only when they are expected in the grammar.
Uname
is used only for function types (Gaussian)
and pseudo-parameters (%f.Area).
Dataset ::= "@"(Digit+|"+"|"*") Varname ::= "$" Lname Funcname ::= "%" Lname QuotedString ::= "'" (AllChars - "'")* "'" Lname ::= (LowerCase | "_") (LowerCase | Digit | "_")* Uname ::= UpperCase AlphaNum+ Number ::= ?number read by strtod()? NonblankString ::= (AllChars - (WhiteSpace | ";" | "#" ))* RestOfLine ::= AllChars*
Fityk library API¶
Fityk comes with embedded Lua interpreter and this language
is used in this section. The API for other supported languages is similar.
Lua communicates with Fityk using object F
of type Fityk
.
To call the methods listed below use F:method()
, for example
F:get_dof()
(not Fityk.get_dof()
).
Note
Other supported languages include C++, C, Python, Perl, Ruby and Java. Except for C, all APIs are similar.
Unlike in built-in Lua, in other cases it is necessary to create
an instance of the Fityk class first. Then you use this object
in the same way as F
is used below.
The fityk.h header file is the best reference.
Additionally, C++ and Python have access to functions from
the ui_api.h header. These functions are used in command line
versions of fityk (cfityk
or its equivalent – samples/cfityk.py
).
Examples of scripts in all the listed languages and in the samples directory.
Here is the most general function:
-
Fityk.
execute
(cmd)¶ Executes a fityk command. Example:
F:execute("fit")
.
The %
operator for the string type is pre-set to support Python-like
formatting:
= "%d pigs" % 3
= "%d %s" % {3, "pigs"}
Input / output¶
-
Fityk.
input
(prompt)¶ Query user. In the CLI user is asked for input in the command line, and in the GUI in a pop-up box. As a special case, if the prompt contains string “[y/n]” the GUI shows Yes/No buttons instead of text entry.
Example: TODO
-
Fityk.
out
(s)¶ Print string in the output area. The
print()
function in built-in Lua is redefined to do the same.
Settings¶
-
Fityk.
set_option_as_string
(opt, val)¶ Set option opt to value val. Equivalent of fityk command
set opt=val
.
-
Fityk.
set_option_as_number
(opt, val)¶ Set option opt to numeric value val.
-
Fityk.
get_option_as_string
(opt)¶ Returns value of opt (string).
-
Fityk.
get_option_as_number
(opt)¶ Returns value of opt (real number).
Data¶
-
Fityk.
load
(spec[, d])¶ Load data to @*d* slot. The first argument is either a string with path or LoadSpec struct that apart from the
path
has also the following optional members:x_col
,y_col
,sig_col
,blocks
,format
,options
. The meaning of these parameters is the same as described in Loading Data.For example, due to limitations of the Fityk DSL a file with the
'
character in the path must be loaded through Lua:lua F:load([[Kat's file.dat]])
LoadSpec is used to specify also format of the file, columns, etc:
spec = fityk.LoadSpec('my.csv') spec.x_col = 1 spec.y_col = 3 spec.format = 'csv' spec.options = 'decimal_comma' F:load(spec)
-
Fityk.
load_data
(d, xx, yy, sigmas[, title])¶ Load data to @*d* slot. xx and yy must be numeric arrays of the same size, sigmas must either be empty or have the same size. title is an optional data title (string).
-
Fityk.
add_point
(x, y, sigma[, d])¶ Add one data point ((x, y) with std. dev. set to sigma) to an existing dataset d. If d is not specified, the default dataset is used.
Example:
F:add_point(30, 7.5, 1)
.
-
Fityk.
get_dataset_count
()¶ Returns number of datasets (n >= 1).
-
Fityk.
get_default_dataset
()¶ Returns default dataset. Default dataset is set by the “use @n” command.
-
Fityk.
get_data
([d])¶ Returns points for dataset d.
in C++ – returns vector<Point>
in Lua – userdata with array-like methods, indexed from 0.
Each point has 4 properties:
x
,y
,sigma
(real numbers) andis_active
(bool).Example:
points = F:get_data() for i = 0, #points-1 do p = points[i] if p.is_active then print(i, p.x, p.y, p.sigma) end end
1 4.24 1.06 1 2 6.73 1.39 1 3 8.8 1.61 1 ...
General Info¶
-
Fityk.
get_info
(s[, d])¶ Returns output of the fityk
info
command as a string. If d is not specified, the default dataset is used (the dataset is relevant for few arguments of theinfo
command).Example:
F:get_info("history")
– returns a multiline string containing all fityk commands issued in this session.
-
Fityk.
calculate_expr
(s[, d])¶ Returns output of the fityk
print
command as a number. If d is not specified, the default dataset is used.Example:
F:calculate_expr("argmax(y)", 0)
.
-
Fityk.
get_view_boundary
(side)¶ Get coordinates of the plotted rectangle, which is set by the
plot
command. Return numeric value corresponding to given side, which should be a letterL
(eft),R
(ight),T
(op) orB
(ottom).
Model info¶
-
Fityk.
get_parameter_count
()¶ Returns number of simple-variables (parameters that can be fitted)
-
Fityk.
all_parameters
()¶ Returns array of simple-variables.
in C++ – vector<double>
in Lua – userdata with array-like methods, indexed from 0.
-
Fityk.
all_variables
()¶ Returns array of all defined variables.
in C++ – vector<Var*>
in Lua – userdata with array-like methods, indexed from 0.
Example:
variables = F:all_variables() for i = 0, #variables-1 do v = variables[i] print(i, v.name, v:value(), v.domain.lo, v.domain.hi, v:gpos(), v:is_simple()) end
Var.is_simple()
returns true for simple-variables.Var.gpos()
returns position of the variable in the global array of parameters (Fityk.all_parameters()), or -1 for compound variables.
-
Fityk.
get_variable
(name)¶ Returns variable $name.
-
Fityk.
all_functions
()¶ Returns array of functions.
in C++ – vector<Func*>
in Lua – userdata with array-like methods, indexed from 0.
Example:
f = F:all_functions()[0] -- first functions print(f.name, f:get_template_name()) -- _1 Gaussian print(f:get_param(0), f:get_param(1)) -- height center print("$" .. f:var_name("height")) -- $_4 print("center:", f:get_param_value("center")) -- center: 24.72235945525 print("f(25)=", f:value_at(25)) -- f(25)= 4386.95533969
-
Fityk.
get_function
(name)¶ Return the function with given name, or NULL if there is no such function.
Example:
f = F:get_function("_1") print("f(25)=", f:value_at(25)) -- f(25)= 4386.95533969
-
Fityk.
get_components
(d[, fz])¶ Returns %functions used in dataset d. If fz is
Z
, returns zero-shift functions.Example:
func = F:get_components(1)[3] -- get 4th (index 3) function in @1 print(func) -- <Func %_6> vname = func:var_name("hwhm") print(vname) -- _21 v = get_variable(vname) print(v, v:value()) -- <Var $_21> 0.1406587
-
Fityk.
get_model_value
(x[, d])¶ Returns the value of the model for dataset
@
d at x.
Fit statistics¶
-
Fityk.
get_wssr
([d])¶ Returns WSSR (weighted sum of squared residuals).
-
Fityk.
get_ssr
([d])¶ Returns SSR (sum of squared residuals).
-
Fityk.
get_rsquared
([d])¶ Returns R-squared.
-
Fityk.
get_dof
([d])¶ Returns the number of degrees of freedom (#points - #parameters).
-
Fityk.
get_covariance_matrix
([d])¶ Returns covariance matrix.
Examples in Lua¶
Show how the peak center moves between datasets:
-- file list-max.lua
prev_x = nil
for n = 0, F:get_dataset_count()-1 do
local path = F:get_info("filename", n)
local filename = string.match(path, "[^/\\]+$") or ""
-- local x = F:calculate_expr("argmax(y)", n)
local x = F:calculate_expr("F[0].center", n)
s = string.format("%s: max at x=%.4f", filename, x)
if prev_x ~= nil then
s = s .. string.format(" (%+.4f)", x-prev_x)
end
prev_x = x
print(s)
end
=-> exec list-max.lua
frame-000.dat: max at x=-0.0197
frame-001.dat: max at x=-0.0209 (-0.0012)
frame-002.dat: max at x=-0.0216 (-0.0007)
frame-003.dat: max at x=-0.0224 (-0.0008)
Write to file values of the model F(x) at chosen x’s (in this example x = 0, 1.5, 3, … 150):
-- file tabular-f.lua
file = io.open("output.dat", "w")
for x = 0, 150, 1.5 do
file:write(string.format("%g\t%g\n", x, F:get_model_value(x)))
end
file:close()
=-> exec tabular-f.lua
=-> !head -5 output.dat
0 12.1761
1.5 12.3004
3 10.9096
4.5 9.12635
6 8.27044