-
Notifications
You must be signed in to change notification settings - Fork 4
This issue was moved to a discussion.
You can continue the conversation there. Go to discussion →
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Which step would one take to integrate an existing Makefile with moulti? #3
Comments
Do you really need
Same: technically, you do not need an intermediate file (but feel free to generate one anyway if that makes sense in your workflow).
I see two main approaches: Approach 2: as you suggest, having markers in the output of
... and turns each title into a Moulti step. That way, Moulti users could "simply" reformat the output of their existing scripts to integrate them with Moulti. This is essentially a less specialized variant of what
Although the quickstart makes you play with two terminals at first, the documentation also details |
I would very much like to avoid doing so, I am happy to read that you think it can be avoided altogether.
This is what I undertood I should do reading the Documentation.md, and it seemed unrealistic to create a whole new makefile system to accomodate for moulti.
See answer for Approach1.
That would be ideal! the moulti instructions could even be hidden behind a "erase current line" terminal command so that it is not visible when running on a system where moulti is not installed. I had a great experience with the klogg app https://klogg.filimonov.dev/ where you can open any log, then search for things and matched lines would list in a window bellow. You could also define regexes and styles to be applied (fg / bg color, bold etc). I guess one direction moulti development could take is follow that approach where a user create some configuration file that corresponds to the way its existing scripts already work, and then wrap the launch of such script. For example, launching make with
Here the steps are started by "Must remake target" and completed by "Successfully remade target", where "[X/Y]" is the progress of such target. If moulti has some DSL to express that in the command line, or with some ad-hoc config file ,it would be easier to integrate with existing machinery. Or I could create a filter that calls moulti appropriately when such strings appear, but I am not even sure how to do that. |
Like this?
#!/usr/bin/env bash
# Instruct the Moulti instance to harvest any output not explicitly assigned to
# a step:
export MOULTI_RUN_OUTPUT=harvest
export MOULTI_INSTANCE="make"
[ "${MOULTI_RUN}" ] || exec moulti run -- "$0" "$@"
function process_lines {
local step_counter=0
while read -r line; do
if [[ "${line}" =~ 'Must remake target' ]]; then
# New target: create a new Moulti step:
local title=$(sed 's/Must remake target `\(.*\)'\''./\1/' <<< "${line}")
((++step_counter))
local step_id="make_$$_${step_counter}"
moulti step add "${step_id}" --title="${title}" --bottom-text=' ' < /dev/null
# Redirect stdout to `moulti pass our_new_step`
exec > >(moulti pass "${step_id}")
fi
[[ "${line}" =~ 'Successfully remade target' ]] && moulti step update "${step_id}" --classes=success
[[ "${line}" =~ error ]] && moulti step update "${step_id}" --classes=error
printf '%s\n' "${line}"
done
}
# Instance title:
moulti set --title="$*"
# Run the command given on the command-line and process each line of its
# output:
"$@" 2>&1 | process_lines Usage: chmod +x moultimake.bash
./moultimake.bash make -d Regarding the rest of your suggestions: thank you, I will take all of this into account when designing the next features. |
It works!!! That’s so cool! I think that example that you just created can easily be adapted to other "filter" use cases, and should be put in the documentation somewhere close to the beginning. :) |
Glad you like it.
It will definitely serve as a basis for my next reflections. Specifically:
|
Yes! And it was easy now with this template to complete with some progress bar \o/ see #!/usr/bin/env bash
# Instruct the Moulti instance to harvest any output not explicitly assigned to
# a step:
export MOULTI_RUN_OUTPUT=harvest
export MOULTI_INSTANCE="make"
[ "${MOULTI_RUN}" ] || exec moulti run -- "$0" "$@"
function process_lines {
local step_counter=0
while read -r line; do
if [[ "${line}" =~ 'Must remake target' ]]; then
# New target: create a new Moulti step:
local title=$(sed 's/Must remake target `\(.*\)'\''./\1/' <<<"${line}")
((++step_counter))
local step_id="make_$$_${step_counter}"
moulti step add "${step_id}" --title="${title}" --bottom-text=' ' </dev/null
# Redirect stdout to `moulti pass our_new_step`
exec > >(moulti pass "${step_id}")
fi
if echo ${line} | grep -q '^\[[0-9]*\/[0-9]*\]'; then
local progress_status=$(echo ${line} | sed -e 's/^\[\([0-9]*\)\/\([0-9]*\).*/\1/')
local progress_target=$(echo ${line} | sed -e 's/^\[\([0-9]*\)\/\([0-9]*\).*/\2/')
moulti set --progress-bar --progress="${progress_status}" --progress-target="${progress_target}"
fi
[[ "${line}" =~ 'Successfully remade target' ]] && moulti step update "${step_id}" --classes=success && moulti set --no-progress-bar
[[ "${line}" =~ error ]] && moulti step update "${step_id}" --classes=error
printf '%s\n' "${line}"
done
}
# Instance title:
moulti set --title="$*"
# Run the command given on the command-line and process each line of its
# output:
"$@" 2>&1 | process_lines Right now, when some steps have happened, I cannot know what happens or happened until I scroll with the mouse over moulti. klogg has a While moulti already scrolls inside steps, could moulti also have something similar to scroll finished/inactive steps ? (should I open new issues?)
That would be my case I guess.
When I see how concise this moultimake.bash is, I wonder if it is necessary at all. Do you have any idea how a declarative file would look like? |
Moulti has programmatic scrolling too. Refer to the documentation for details but TL;DR:
|
That will be part of my reflections too.
The default approach would be a serializable Python dict written as JSON/YAML/TOML that describes:
+ support for using regex captures and environment variables in actions. I could implement that by myself OR I could look for a tool/library/framework that already provides this kind of logic. |
That’s very neat! I wonder how I missed that when searching in the documentation. Thanks!
I tried those, but I have troubles understanding if scroll stops, or if scripts takes time. I will use
Reading all of this, I wonder if a python API might be easier to implement (and more powerful) than JSON/YAML/TMOL. This api would read like declarative, but really just call python functions, a bit like the cmake syntax. import moulti as m
m.detector.add_new_step_rule("Must remake target `\(.*\)'\'", auto_increment_id=True, show_progress_bar=False)
m.detector.add_step_success_rule(".*Successfully remade target.*")
m.detector.add_step_error_rule(".*error.*")
m.detector.add_step_status_progress_rule("^\[\([0-9]*\)\/[0-9]*.*", force_show_progress_bar=True)
m.detector.add_step_total_progress_rule("^\[[0-9]*\/\([0-9]*\).*", force_show_progress_bar=True)
m.add_action(["make", "-d", "configure"], can_be_restarted=True, on_success="next_command")
m.add_action(["make", "-d", "build"], can_be_restarted=True, on_success="next_command")
m.add_action(["make", "-d", "test"], can_be_restarted=True)
m.run(run_output="harvest", instance="make", start_instance_if_needed=True, follow=True) Later part can easily be made customizable using vanilla python: for i in range(1,len(argv)):
m.add_restartable_command(["make", "-d", argv[i)], can_be_restarted=True, on_success="next_command")
m.run(run_output="harvest", instance="make", start_instance_if_needed=True, follow=True) |
They are complementary features: |
It is planned to implement a "Python User API", i.e. a Python API exposed to users and that would adhere to semver. It will definitely be more powerful (e.g. it should integrate a class that keeps track of all created steps, and it should not need as many fork-exec-connect-send-receive-close-exit operations as a shell script) but this will likely wait until things get more stable (because a stable feature set makes it easier to design a perennial API). My reflections so far:
By the way, real life use cases are most welcome to feed those reflections. Do you have any other use case in mind? |
This issue was moved to a discussion.
You can continue the conversation there. Go to discussion →
Hello,
Thanks for creating this software!
Which step would one take to integrate an existing Makefile with moulti?
One possibility might be to use
script scrip.txt make configure build
to get the make output to go to a text fail, thentail
that file to moulti.However, how to differentiate Makefile steps ? Should I instrument the makefile to emit markers ?
Also, how to do all that in the same terminal instead of multiple ones?
Best,
The text was updated successfully, but these errors were encountered: