Several gadgets for Linux. No warranties are for the listed tools.
Trick for target scheduling.
- GNU bash 4.4.19(1)-release
- Python 3.6.7
- Vi IMproved 8.0, with patch 1-1766
target init
- initiate a target pool (~/.target_pool
)target edit
- edit the target pooltarget plan
- edit the every-week plan (~/.target_update_plan
)target ddl
- edit the one-shot DDL-s (~/.target_ddl
)target update
- update the weights in the target pool in accordance with~/.target_update_plan
and~/.target_ddl
; this command will update the weights subject to a reciprocal function of the difference to the ddltarget select
- randomly select a target in accordance with the weights in the target pooltarget list-ddls
- list the closest DDL-s in~/.target_update_plan
and~/.target_ddl
In order to configure bash
to autocomplete the command names, add following statements to ~/.bashrc
.
function _target() {
COMPREPLY=($(compgen -W "update edit init select plan ddl list-ddls" ${COMP_WORDS[$COMP_CWORD]}))
return 0
}
complete -F _target target
The target pool file is in the following form:
# target weight
Read papers 3
Lab 4
Literary read 2
The lines starting with "#" will be considered as comments. The last field separated by white space characters is considered as the weight while the other fields are considered as the target name integrally.
The plan file is in the following form:
# target DDL (weekday)
Application for summer practice 4
Prepare for experiment 1
The lines starting with "#" will be considered as comments as well. The last field separated by white space characters is considered as the weekday of the periodic deadline of the particular target while the other fields are considered as the target name integrally as well.
The ddl file is in the similar form with the pool file and the plan file:
# target DDL (yyyy-mm-dd)
Big project of Intelligent Optimisation Algorithms 2019-6-13
Report of oral history requested for Mao Thought 2019-5-31
The difference is that the last field indicates the one-shot ddl of the corresponding target but not the weight or the periodic ddl.
An external extension like fcitx.vim
plugin which helps to activate or deactivate fcitx5
automatically on entering or leaving the insertion mode when you're using vim
on a remote machine through ssh
. This tool isn't plugged into vim
, however, rather than fcitx.vim
works as a plugin for vim
. A daemon should be started on the remote machine. The client on the local machine connects to the remote daemon and receives the status of the remote vim
from the daemon and switch the status of fcitx5
accordingly.
Local machine:
- Manjaro 20.1 Mikah
- x86_64 Linux 5.7.15-1-MANJARO
- KDE 5.73.0 / Plasma 5.19.4
- fcitx5 4.99.0
- Python 3.8.5
- openssh 8.3p1
Remote machine:
- Ubuntu 18.04 bionic
- x86_64 Linux 4.15.0-109-generic
- Vi IMproved 8.0, with patches 1-1453
- Python 3.6.9
- openssh-server 1:7.6p1
After ssh-ing to the remote host, start the daemon. The daemon will be bound to 0.0.0.0:30002
by default. The default address and port could be modified by
python3 remote_fcitx_vim_daemon.py --address ADDRESS --port PORT &>>/dev/null &
Then launch the local client and connect to the remote daemon.
python3 remote_fcitx_vim_local_machine.py [--port REMOTE_PORT] REMOTE_ADDRESS &>>/dev/null &
Add the content in the for_vimrc_on_server
to ~/.vimrc
and replace the value of s:remote_fcitx_vim_path
with the path to remote_fcitx_vim_sender.py
.
And you could use this extension now. Enjoy it!
When the remote host is behind an NAT, maybe you cannot connect to the port used by the underlying daemon directly. In this case, you may need to configure a new port mapping on the remote gateway or traverse the NAT by some way.
If you have no idea to traverse the NAT or you don't want to install any additional software, here is an alternative solution. SSH2 protocol provides a simple forwarding and proxy function. Simply login to the remote machine with -L
option:
ssh -L <local_port>:<daemon_address>:<daemon_port> -p <remote_port> remote_user@remote_host
The connection to the <local_port>
will be forwarded to <daemon_address>:<daemon_port>
automatically through the ssh connection. As for this extension, since the daemon is directly run on the ssh server, <daemon_address>
could be simply set to 127.0.0.1
. <daemon_address>
is the port the daemon listens to (30002
by default).
For more details w.r.t. the usage of ssh -L
and ssh -R
, please refer to the man page of ssh
.
Note that in the current version, unix domain socket is adopted for "Socket B" on the figure. It could be replaced by an internet socket so as to support the fcitx
status management of multiple hosts with singular daemon.
Timestamp your standard output!
- x86_64 Linux 5.14.7-2-MANJARO
- GNU bash,版本 5.1.8(1)-release (x86_64-pc-linux-gnu)
- Python 3.9.7
There are two ways to use the shell timestamper. One way is to wrap the target command with the tstamp
instruction, just like the usage of time
instruction in bash.
tstamp [--format FORMAT] [--buffer-mode BUFFER_MODE] -- command ...
Besides, tstamp
could be appended behind the target command through a pipe, just like the usage of tee
.
command ... |tstamp [--format FORMAT] [--buffer-mode BUFFER_MODE]
--format
option could be abbreviated as -f
and --buffer-mode
- -m
. --format
controls the format of the timestamp. The format string complys the protocol in Python 3 datetime
module and defaults to %Y-%m-%d %H:%M:%S.%f
. --buffer-mode
controls the buffer mode, for which two choices are availabe: c
and L
. c
instructs the stamper to flush the buffer per character, or just say, use no buffer. L
instructs the stamper to flush the buffer per line, which is the default behaviour for most command line programs. However, the default mode of stamper is set to c
for the best support to the programs requiring standard inputs. If your program requires no user's input and the standard I/O performance is crucial, it is recommended to use the L
mode.
A simple tool helping manual version control.
- x86_64 Linux 5.15.25-1-MANJARO
- GNU bash,版本 5.1.16(1)-release (x86_64-pc-linux-gnu)
- Python 3.10.2
version --help
usage: version [-h] {init,list,check,commit,checkout,log,export} ...
positional arguments:
{init,list,check,commit,checkout,log,export}
init Initiate a version repository.
list List tracked files.
check Check the versions of a specific file.
commit Commit a file version.
checkout Checkout a file version
log Print the daily log.
export Export the version table.
options:
-h, --help show this help message and exit
Check the help output for the subcommands like version checkout --help
for detailed usage.
function _complete_version() {
local subcommands="init list check commit checkout log export"
if [[ $COMP_CWORD -le 2 ]]; then
COMPREPLY=($(compgen -W "$subcommands" ${COMP_WORDS[$COMP_CWORD]}))
fi
}
complete -F _complete_version version
- x86_64 Linux 5.15.28-1-MANJARO
- GNU bash,版本 5.1.16(1)-release (x86_64-pc-linux-gnu)
- Python 3.10.2
zpp --help
usage: zpp [-h] [--prefix PREFIX] [--suffix SUFFIX] [--nosuffix] [--mode {H,T,C,J}] [--def MACRO] [--output OUTPUT] file
positional arguments:
file Input file.
options:
-h, --help show this help message and exit
--prefix PREFIX Prefix for preprocessing commands.
--suffix SUFFIX Suffix for preprocessing commands.
--nosuffix No suffices is in need.
--mode {H,T,C,J}, -m {H,T,C,J}
H: HTML Comment Mode, e.g., <!-- include a.js -->
T: TeX Comment Mode, e.g., % define TeX \LaTeX
C: C Preprocessing Instruction Mode, e.g., #define A_TOY_PREPROCESSOR
J: Java Comment Mode, e.g., // include class.java
--def MACRO Manually define a macro like "ABC" or "ABC=LSP", "=" in macro name and definitions could be escaped by "\"
--output OUTPUT, -o OUTPUT
Output file
Lines for the preprocessing commands are recognized by the user-specific prefix
and suffix. For convenience, the tool offers four predefined modes with
specific prefix and suffix. The default mode is H
- the HTML Comment Mode
with prefix <!--
and suffix -->
.
include
include [-PREFIX] [+SUFFIX] [<LINEPREFIX<] [>LINESUFFIX>] [/REGEX/SUBSTITUTION/] FILENAME
Inserts the contents from FILENAME
at the current position. The preprocessing
commands in the included file will be handled as well.
A different command prefix or suffix could be specified for the included file.
For example, the main file is preprocessed under H
mode, i.e., with prefix
<!--
and suffix -->
. Now I want to include a file foo.ctxt
with C-style
preprocessing command prefix, I can use include
command as:
<!-- include -# + foo.ctxt -->
The command also supports to specify a prefix and a suffix for all the included
lines, i.e., the included lines will be prepended or appended with the
specified line prefix/suffix when inserting into the referencing file. For
instance, I'm writing a README in markdown. I want to include the contents from
another file b.md
and make the included contents into quotations, I can use
include
command as:
<!-- include <> < b.md -->
Then, all the inserted contents will be prefixed a >
so that they will be
rendered as quotations.
Also text substituion is allowed when processing file including. The regex
syntax follows re
module of
Python 3.
Normally, a relative path is expected. However, you can also specify an
absolute path by prefixing FILENAME
with a #
. For instance,
% include /None/\?/ #/home/foo/some/project/some/file
define
andundef
define MACRO_NAME [MACRO_VALUE]
Defines a macro which will be replaced in the following contents. If it is
omitted, MACRO_VALUE
will default to an empty string. During substitution,
the MACRO_NAME
is matched with ASCII word boundaries.
If more sophiscated MACRO_NAME
match is expected, the following command
should be used:
defineR MACRO_NAME /REGEX/[FLAGS] SUBSTITUTION
MACRO_NAME
can be used in if commands and SUBSTITUTION
is considered the
original MACRO_VALUE
. Regex flags are optional. If present, each flag should
comprise only one letter.
Macros can also be defined through command line arguments, the syntax is:
zpp --def MACRO_NAME1=MACRO_VALUE --def MACRO_NAME2=/REGEX/SUBSTITUTION/ input -o output
undef MACRO_NAME
revokes the definition of an existing macro.
- if commands
ifdef MACRO_NAME
ifndef MACRO_NAME
ifeq MACRO_NAME MACRO_VALUE
ifneq MACRO_NAME MACRO_VALUE
ifeqn MACRO_NAME MACRO_VALUE
ifneqn MACRO_NAME MACRO_VALUE
ifge MACRO_NAME MACRO_VALUE
ifnge MACRO_NAME MACRO_VALUE
ifle MACRO_NAME MACRO_VALUE
ifnle MACRO_NAME MACRO_VALUE
ifgt MACRO_NAME MACRO_VALUE
ifngt MACRO_NAME MACRO_VALUE
iflt MACRO_NAME MACRO_VALUE
ifnlt MACRO_NAME MACRO_VALUE
elifdef MACRO_NAME
elifndef MACRO_NAME
elifeq MACRO_NAME MACRO_VALUE
elifneq MACRO_NAME MACRO_VALUE
def
checks if a macro is already defined and eq
checks if a macro is
already defined and if the macro definition equals to the given value. eqn
,
ge
, le
, gt
, and lt
will compare the values in number. n
will reverse
the check result. However, note that if a macro is not defined, the n*
(i.e., neq
, neqn
, nge
, etc) will lead to false which is the same as
eq
and eqn
.
if
commands starts a if block and elif
commands give another choice.
Also else
command is supported.
endif
is used to end an if block.
Embedding if
blocks is supported.
- Manjaro 22.0.0 Sikaris
- KDE 5.98.0 / Plasma 5.25.5
- Python 3.10.7
- Mozilla Firefox 106.0.2
- wmctrl 1.07
- xprop 1.2.5
This extension will move it to the firefox window in the current kde activity when a new tab is opened. Currently it only supports on X.
This extension contains a plug-in for firefox and a external Python script. The
plug-in could be found in Mozilla extension store
by kde-activities-z
. The external Python should be installed in the user's
$PATH and a manifest json for the external script is required to be place at
~/.mozilla/native-messaging-hosts
and the path
field in it should be set to
the path to the installed script. In convenience, a makefile is provided and a
simple make
command should install the script to ~/.local/bin
and complete
the configurations.
Owing to the limit of the API of firefox, neither the external script can obtain the correct current activity where there havn't been any firefox windows, nor the plug-in js can open a new window in the current activity without any firefox windows in it. I'v got no ideas for this issue temporarily.
The external script first revokes wmctrl -lx
to obtain all the opened X
windows and filters out the firefox instances. Then,
qdbus org.kde.ActivityManager /ActivityManager/Activities CurrentActivity
is revoked to get the current activity id. xprop -id $wid _KDE_NET_WM_ACTIVITIES
could be used to check the activity which a specific window is on. By this way,
the script can determine the correct firefox window to which the new tab should
be moved.
This is a template tool to work with LLM prompts. It utilize zpp
for text
preprocessing.
from prompt_template import TemplateGroup, PromptGroupT
from PIL import Image
import openai
template: TemplateGroup = TemplateGroup.parse("main.prompt")
prompt: PromptGroupT = template.safe_substitute( text_mapping={"slot": "value"}
, img_mapping={"slot": Image.open("a.png")}
)
completor = openai.OpenAI(api_key="sk-xxx")
completion = completor.chat.completions.create( model="gpt-3.5-turbo-1106"
, messages=prompt
)
This tool supports the same format of placeholders as the default format of
Python's string.Template
. You can use $slot
or ${slot}
to specify a text
placeholder and use $$
to specify a literal $
.
This tool also supports image placeholders. You can use $image_slot
or
${image_slot}
to specify a image placeholder. Note the placeholder is
supposed to reference to in img_mapping
argument of safe_substitute
method
without the image_
prefix, i.e., you should use slot
to reference to
placeholder $image_slot
. You can also use ${imagef:const.jpg}
to specify an
image constant. Optional resizing can be declared by specifying the new width
and height ${imagef:const.jpg:1080:1920}
.
%%% chat
declares the target prompt format, typically chat completion (chat
) or text
completion (instruct
).
---system
specifies the role of the following contents until meeting the next ---role
.
The roles used by OpenAI commands are system
, user
, and assistant
.
### slot
Contents
###
defines an optional default value for text placeholders. Note that the last
\n
of Contents
(i.e., before the ending ###
) is ignored during
substituion.
### image_slot
${imagef:default.png:300:500}
###
defines optional default image for image placeholders.
% comments
Such lines are considered comments.
\\\%%% NOTE! This is a literal line
When you want to use several lines which will be confused with the special meta
lines in the prompt, you can precede the line with \\\
to escapte it.
``` snippet_name
snippet contents
```
declares a snippet that can be re-used. Multiple candidates can be declared for
a snippet seperated by a line containing literally ---
. You can control which
candidate to be used in instantiation by fix_snippet_choice
parameter of
safe_substitute
method. By default, a random candidate is sampled for each
instantiation.
=== snippet_name instance_id
instantiate a declared snippet. To distinguish the placeholders in different
instances of a snippet, you can reference the placeholder by the name declared
in the snippet suffixing the instance_id
, both in safe_substitute
method
and ###
default value declaration, both for text and image plachoders. For
instance, the following snippet defines two placeholders: ${slot}
and
${image_slot}
:
``` snippet
Text placeholder: ${slot}
Image placeholder: ${image_slot}
```
And we create an instance:
=== snippet id
Then you can use following commands to declare default value for this instance
### slot_id
default text
###
### image_slot_id
${imagef:default_image.png}
###
or substitue placeholders by
template.safe_substitute( text_mapping={"slot_id": "running text"}
, img_mapping={"slot_id": running_image}
)
If you don't want to declare default values for each instance separately, you can also declare a common default value like:
### slot
common default text
###
Snippet instances can be used in common templates or in other latter snippet definitions. Recursive instantiation is not supported.
Zpp preprocessing commands like include
, define
,
ifdef
, etc. are also supported. This tool adopts the TeX mode by default.
An example and a syntax Highlighting file for vim is offered under custom configs/syntax
.
This tool uses dbus-monitor to monitor the activity switch on KDE Plasma desktops and execute scripts automatically.
- Manjaro 24.0.7 Wynsdey
- KDE 6.5.0 / Plasma 6.0.5
- X.Org version: 21.1.13
- Kernel: x86_64 Linux 6.6.46-1-MANJARO
- dbus-broker 36
- dbus 1.14.10
- qt5-tools 5.15.14+kde+r4
- bash 5.2.32
- GNU Awk 5.3.0, API 4.0, PMA Avon 8-g1, (GNU MPFR 4.2.1, GNU MP 6.3.0)
- systemd 256.4
The scripts to be executed are supposed to be stored under
~/.autoactions/currentActivityChanged
. When an activity is left, suppose its
id is "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx", the tool will automatically
invokes
~/.autoactions/currentActivityChanged/exit=xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx
if it exists and is executable. When an activity is entered, suppose its id is
"yyyyyyyy-yyyy-yyyy-yyyy-yyyyyyyyyyyy", the tool will automatically invokes
~/.autoactions/currentActivityChanged/exit=yyyyyyyy-yyyy-yyyy-yyyy-yyyyyyyyyyyy
if it exists and is executable.
The entering and exiting scripts can also be specified by activity name like
name=workspace
and exit-name=workspace
.
~/.sogoubackup/backup
- backup the configs of Sogou Input Method routinelyconvert_file_name
- convert the extension name of the images from the mobile from 'jpeg' to the correct oneeasy-backup-tool
- rsync configuration for directory backupeasy-snapshots
- rsync configuration for directory snapshots添加元信息.ptsp
- script for picardbranch-tool
- snapshot the workspace for multi-branch experiment directoryxclip_image_saver
- quick save the image base64 in clipboard to an image file
Corresponding completion configuration:
function _complete_backup() {
if [[ $COMP_CWORD -le 2 ]]; then
COMPREPLY=($(compgen -W "init backup" ${COMP_WORDS[$COMP_CWORD]}))
fi
}
complete -F _complete_backup bckp
complete -F _complete_backup zsnap
A simple encryption tool using XOR algorithm.
- Python 3.6.7
encrypt --help
usage: encrypt [-h] [--password [PASSWORD]] [--decode] [--delete]
[--recursive] [--suffix SUFFIX]
input [input ...]
positional arguments:
input The file to encrypt or decrypt
optional arguments:
-h, --help show this help message and exit
--password [PASSWORD], -p [PASSWORD]
Indicates whether a password should be used
--decode, -d Decrypts the file if set
--delete Delete the inputed file if set
--recursive, -r Encrypt the files under the directory recursively if
set. With "--delete" option set implicitly
--suffix SUFFIX, -a SUFFIX
Specify the suffix of the encrypted file. The suffix
will be appended to the inputed file name during
encryption and be truncated from the inputed file name
during decryption
Base64 decoder constructed in Python 3.
- Python 3.6.7
base64decode --help
usage: base64decode [-h] [--decode DECODE] [--file FILE] [--encoding ENCODING]
optional arguments:
-h, --help show this help message and exit
--decode DECODE, -d DECODE
Base64 string to be decoded
--file FILE, -f FILE Text file where stored Base64 string to be decoded; "-" for stdin; If either
this option or "--decode" isn't set, the coded string will be read from stdin
--encoding ENCODING Encoding of decoded string; Default to utf-8
Now such config files as follows are collected in this repo:
~/.bashrc
~/.profile
~/.inputrc
~/.vimrc
~/.vim/ftdetect
~/.vim/syntax
~/.tmux.conf
~/.config/fcitx/data/punc-ng.mb.zh_CN
~/.local/share/fcitx5/data/QuickPhrase.mb
(~/.config/fcitx/data/QuickPhrase.mb
)~/.config/htop/htoprc
~/.gotop/layout
Under directory w3m
:
~/.w3m/keymap
Under directory terminator
:
~/.config/terminator/config
~/.config/systemd/user/rem_schedule.service
- forremind
daemon~/.config/systemd/user/remote-fcitx-daemon.service
- forremote-fcitx-vim
daemon~/.config/systemd/user/activity_switch_watcher.service
- watching KDE activity switch and do actions