mirror of
https://github.com/crazywhalecc/quick-shell.git
synced 2026-03-17 20:34:52 +08:00
Compare commits
15 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
175c56e63b | ||
|
|
d06cab0035 | ||
|
|
8c61694893 | ||
|
|
207164aa6c | ||
|
|
d2b14780e7 | ||
|
|
f262cf826d | ||
|
|
d7d39c2903 | ||
|
|
371ddb084d | ||
|
|
80f3b94fff | ||
|
|
632e03e041 | ||
|
|
d2808f94be | ||
|
|
d26c0d2ded | ||
|
|
7bec18d525 | ||
|
|
c5a16fe3aa | ||
|
|
35401bbade |
23
README.md
23
README.md
@ -1,11 +1,11 @@
|
||||
# quick-shell
|
||||
|
||||
一个通过 `curl`、管道、`bash` 实现的快速执行常用命令的工具,不在本地硬盘保存脚本。
|
||||
一个通过 `curl`、管道和 `bash` 实现的快速执行常用命令的工具,用于 Linux。
|
||||
|
||||
万能入口:
|
||||
|
||||
```bash
|
||||
bash <(curl -s http://shell.zhamao.xin/)
|
||||
bash <(curl -s https://shell.zhamao.xin/)
|
||||
```
|
||||
|
||||
## 用法
|
||||
@ -14,16 +14,19 @@ bash <(curl -s http://shell.zhamao.xin/)
|
||||
|
||||
```bash
|
||||
# 快速运行neofetch
|
||||
bash <(curl -s http://shell.zhamao.xin/neofetch)
|
||||
bash <(curl -s https://shell.zhamao.xin/neofetch)
|
||||
|
||||
# 快速运行speedtest
|
||||
bash <(curl -s http://shell.zhamao.xin/speedtest)
|
||||
bash <(curl -s https://shell.zhamao.xin/speedtest)
|
||||
|
||||
# 在终端用方向键玩 2048
|
||||
bash <(curl -s https://shell.zhamao.xin/game/2048)
|
||||
|
||||
# 使用CTF工具箱(比如快速开启frp内网穿透)
|
||||
# 服务端,监听0.0.0.0:7001,假设我的公网服务器IP是1.2.3.4
|
||||
bash <(curl -s http://shell.zhamao.xin/ctf/frps/bind_addr=0.0.0.0/bind_port=7001)
|
||||
bash <(curl -s https://shell.zhamao.xin/ctf/frps/bind_addr=0.0.0.0/bind_port=7001)
|
||||
# 客户端,穿透ssh 22端口到公网的50022端口
|
||||
bash <(curl -s http://shell.zhamao.xin/ctf/frpc/1.2.3.4:7001/127.0.0.1/22/50022)
|
||||
bash <(curl -s https://shell.zhamao.xin/ctf/frpc/1.2.3.4:7001/127.0.0.1/22/50022)
|
||||
```
|
||||
|
||||
## 支持的快速命令
|
||||
@ -36,8 +39,14 @@ bash <(curl -s http://shell.zhamao.xin/ctf/frpc/1.2.3.4:7001/127.0.0.1/22/50022)
|
||||
| `ctf/reverse_shell` | 使用 bash 反弹 shell,提供一个目标的 IP 和 nc 端口即可 |
|
||||
| `ctf/frps` | 快速启动一个 frps 内网穿透服务器 |
|
||||
| `ctf/frpc` | 快速使用 frpc 代理一个内网穿透一个端口,提供一个目标的 IP 和 TCP 端口即可 |
|
||||
| `php_server` | 快速启动一个当前目录下的 Web 服务器(支持 PHP 脚本) |
|
||||
| `game/2048` | 在线游玩 2048 小游戏(终端) |
|
||||
| ... | 持续更新中 |
|
||||
|
||||
## 安全
|
||||
|
||||
本项目所有的指令如果不放心直接执行可以先 curl 进行查看,也可以使用 `showcode/{name}` 查看对应命令的生成源码。例如 `bash <(curl -s https://shell.zhamao.xin/showcode/ctf/frpc)`。
|
||||
|
||||
## 长啥样
|
||||
|
||||
<img width="700" alt="未命名_副本" src="https://user-images.githubusercontent.com/20330940/170800524-0ced4fdb-97b0-4961-b281-3c96af334095.png">
|
||||
@ -46,6 +55,6 @@ bash <(curl -s http://shell.zhamao.xin/ctf/frpc/1.2.3.4:7001/127.0.0.1/22/50022)
|
||||
|
||||
## 贡献
|
||||
|
||||
如果你觉得有常用的命令,可以提交到这个项目中,我会尽快添加到这个项目中。
|
||||
如果你觉得有常用的命令,可以提交到这个项目中,我会尽快添加到这个项目。
|
||||
|
||||
目前这个项目目前只是自己用,如果觉得不好,请提出指导性意见!
|
||||
|
||||
@ -3,6 +3,7 @@
|
||||
namespace QuickShell\Commands;
|
||||
|
||||
use QuickShell\Annotations\Command;
|
||||
use QuickShell\Annotations\CommandOption;
|
||||
|
||||
class ExampleCommand
|
||||
{
|
||||
@ -17,4 +18,33 @@ class ExampleCommand
|
||||
{
|
||||
return cmd("curl -s http://ip.zhamao.xin");
|
||||
}
|
||||
|
||||
#[Command('php_server', '在当前目录使用PHP启动一个Web服务器')]
|
||||
#[CommandOption(option_name: 'port', description: '端口号', required: true)]
|
||||
public function phpServer(array $params)
|
||||
{
|
||||
$cmd = <<<CMD
|
||||
case \$(uname -s) in
|
||||
Linux) mysys="linux" ;;
|
||||
*)
|
||||
echo "Only support Linux!"
|
||||
exit 1
|
||||
;;
|
||||
esac
|
||||
if [ ! -f "/tmp/.qs_php" ]; then
|
||||
echo "sys: \$mysys"
|
||||
link="https://dl.zhamao.xin/php-bin/down.php?php_ver=8.1&arch=\$(uname -m)"
|
||||
echo "Downloading php from \$link"
|
||||
curl \$link -o /tmp/php.tgz -L && \
|
||||
cd /tmp && \
|
||||
tar -xzvf php.tgz && \
|
||||
rm php.tgz && \
|
||||
mv php .qs_php && \
|
||||
chmod +x .qs_php
|
||||
fi
|
||||
/tmp/.qs_php -S 0.0.0.0:{port}
|
||||
CMD;
|
||||
$cmd = str_replace('{port}', $params['port'] ?? '8080', $cmd);
|
||||
return cmd($cmd);
|
||||
}
|
||||
}
|
||||
385
src/QuickShell/Commands/GameCommand.php
Normal file
385
src/QuickShell/Commands/GameCommand.php
Normal file
@ -0,0 +1,385 @@
|
||||
<?php
|
||||
|
||||
namespace QuickShell\Commands;
|
||||
|
||||
use QuickShell\Annotations\Command;
|
||||
use QuickShell\Annotations\CommandCategory;
|
||||
|
||||
#[CommandCategory('game', '游戏')]
|
||||
class GameCommand
|
||||
{
|
||||
#[Command('2048', '在线游玩2048游戏')]
|
||||
public function run2048()
|
||||
{
|
||||
$cmd = <<<CMD
|
||||
#!/usr/bin/env bash
|
||||
|
||||
#important variables
|
||||
declare -ia board # array that keeps track of game status
|
||||
declare -i pieces # number of pieces present on board
|
||||
declare -i score=0 # score variable
|
||||
declare -i flag_skip # flag that prevents doing more than one operation on
|
||||
# single field in one step
|
||||
declare -i moves # stores number of possible moves to determine if player lost
|
||||
# the game
|
||||
declare ESC=\$'\e' # escape byte
|
||||
declare header="Bash 2048 v1.1 (https://github.com/mydzor/bash2048) CtrlC to stop |"
|
||||
|
||||
declare -i start_time=\$(date +%s)
|
||||
|
||||
#default config
|
||||
declare -i board_size=4
|
||||
declare -i target=2048
|
||||
declare -i reload_flag=0
|
||||
declare config_dir="/tmp/.bash2048"
|
||||
|
||||
#for colorizing numbers
|
||||
declare -a colors
|
||||
colors[2]=33 # yellow text
|
||||
colors[4]=32 # green text
|
||||
colors[8]=34 # blue text
|
||||
colors[16]=36 # cyan text
|
||||
colors[32]=35 # purple text
|
||||
colors[64]="33m\033[7" # yellow background
|
||||
colors[128]="32m\033[7" # green background
|
||||
colors[256]="34m\033[7" # blue background
|
||||
colors[512]="36m\033[7" # cyan background
|
||||
colors[1024]="35m\033[7" # purple background
|
||||
colors[2048]="31m\033[7" # red background (won with default target)
|
||||
|
||||
exec 3>/dev/null # no logging by default
|
||||
|
||||
trap "end_game 0 1" INT #handle INT signal
|
||||
|
||||
#simplified replacement of seq command
|
||||
function _seq {
|
||||
local cur=1
|
||||
local max
|
||||
local inc=1
|
||||
case \$# in
|
||||
1) let max=\$1;;
|
||||
2) let cur=\$1
|
||||
let max=\$2;;
|
||||
3) let cur=\$1
|
||||
let inc=\$2
|
||||
let max=\$3;;
|
||||
esac
|
||||
while test \$max -ge \$cur; do
|
||||
printf "\$cur "
|
||||
let cur+=inc
|
||||
done
|
||||
}
|
||||
|
||||
# print currect status of the game, last added pieces are marked red
|
||||
function print_board {
|
||||
clear
|
||||
printf "\$header pieces=\$pieces target=\$target score=\$score\n"
|
||||
printf "Board status:\n" >&3
|
||||
printf "\n"
|
||||
printf '/------'
|
||||
for l in \$(_seq 1 \$index_max); do
|
||||
printf '+------'
|
||||
done
|
||||
printf '\\\n'
|
||||
for l in \$(_seq 0 \$index_max); do
|
||||
printf '|'
|
||||
for m in \$(_seq 0 \$index_max); do
|
||||
if let \${board[l*\$board_size+m]}; then
|
||||
if let '(last_added==(l*board_size+m))|(first_round==(l*board_size+m))'; then
|
||||
printf '\033[1m\033[31m %4d \033[0m|' \${board[l*\$board_size+m]}
|
||||
else
|
||||
printf "\033[1m\033[\${colors[\${board[l*\$board_size+m]}]}m %4d\033[0m |" \${board[l*\$board_size+m]}
|
||||
fi
|
||||
printf " %4d |" \${board[l*\$board_size+m]} >&3
|
||||
else
|
||||
printf ' |'
|
||||
printf ' |' >&3
|
||||
fi
|
||||
done
|
||||
let l==\$index_max || {
|
||||
printf '\n|------'
|
||||
for l in \$(_seq 1 \$index_max); do
|
||||
printf '+------'
|
||||
done
|
||||
printf '|\n'
|
||||
printf '\n' >&3
|
||||
}
|
||||
done
|
||||
printf '\n\\------'
|
||||
for l in \$(_seq 1 \$index_max); do
|
||||
printf '+------'
|
||||
done
|
||||
printf '/\n'
|
||||
}
|
||||
|
||||
# Generate new piece on the board
|
||||
# inputs:
|
||||
# \$board - original state of the game board
|
||||
# \$pieces - original number of pieces
|
||||
# outputs:
|
||||
# \$board - new state of the game board
|
||||
# \$pieces - new number of pieces
|
||||
function generate_piece {
|
||||
while true; do
|
||||
let pos=RANDOM%fields_total
|
||||
let board[\$pos] || {
|
||||
let value=RANDOM%10?2:4
|
||||
board[\$pos]=\$value
|
||||
last_added=\$pos
|
||||
printf "Generated new piece with value \$value at position [\$pos]\n" >&3
|
||||
break;
|
||||
}
|
||||
done
|
||||
let pieces++
|
||||
}
|
||||
|
||||
# perform push operation between two pieces
|
||||
# inputs:
|
||||
# \$1 - push position, for horizontal push this is row, for vertical column
|
||||
# \$2 - recipient piece, this will hold result if moving or joining
|
||||
# \$3 - originator piece, after moving or joining this will be left empty
|
||||
# \$4 - direction of push, can be either "up", "down", "left" or "right"
|
||||
# \$5 - if anything is passed, do not perform the push, only update number
|
||||
# of valid moves
|
||||
# \$board - original state of the game board
|
||||
# outputs:
|
||||
# \$change - indicates if the board was changed this round
|
||||
# \$flag_skip - indicates that recipient piece cannot be modified further
|
||||
# \$board - new state of the game board
|
||||
function push_pieces {
|
||||
case \$4 in
|
||||
"up")
|
||||
let "first=\$2*\$board_size+\$1"
|
||||
let "second=(\$2+\$3)*\$board_size+\$1"
|
||||
;;
|
||||
"down")
|
||||
let "first=(index_max-\$2)*\$board_size+\$1"
|
||||
let "second=(index_max-\$2-\$3)*\$board_size+\$1"
|
||||
;;
|
||||
"left")
|
||||
let "first=\$1*\$board_size+\$2"
|
||||
let "second=\$1*\$board_size+(\$2+\$3)"
|
||||
;;
|
||||
"right")
|
||||
let "first=\$1*\$board_size+(index_max-\$2)"
|
||||
let "second=\$1*\$board_size+(index_max-\$2-\$3)"
|
||||
;;
|
||||
esac
|
||||
let \${board[\$first]} || {
|
||||
let \${board[\$second]} && {
|
||||
if test -z \$5; then
|
||||
board[\$first]=\${board[\$second]}
|
||||
let board[\$second]=0
|
||||
let change=1
|
||||
printf "move piece with value \${board[\$first]} from [\$second] to [\$first]\n" >&3
|
||||
else
|
||||
let moves++
|
||||
fi
|
||||
return
|
||||
}
|
||||
return
|
||||
}
|
||||
let \${board[\$second]} && let flag_skip=1
|
||||
let "\${board[\$first]}==\${board[second]}" && {
|
||||
if test -z \$5; then
|
||||
let board[\$first]*=2
|
||||
let "board[\$first]==\$target" && end_game 1
|
||||
let board[\$second]=0
|
||||
let pieces-=1
|
||||
let change=1
|
||||
let score+=\${board[\$first]}
|
||||
printf "joined piece from [\$second] with [\$first], new value=\${board[\$first]}\n" >&3
|
||||
else
|
||||
let moves++
|
||||
fi
|
||||
}
|
||||
}
|
||||
|
||||
function apply_push {
|
||||
printf "\n\ninput: \$1 key\n" >&3
|
||||
for i in \$(_seq 0 \$index_max); do
|
||||
for j in \$(_seq 0 \$index_max); do
|
||||
flag_skip=0
|
||||
let increment_max=index_max-j
|
||||
for k in \$(_seq 1 \$increment_max); do
|
||||
let flag_skip && break
|
||||
push_pieces \$i \$j \$k \$1 \$2
|
||||
done
|
||||
done
|
||||
done
|
||||
}
|
||||
|
||||
function check_moves {
|
||||
let moves=0
|
||||
apply_push up fake
|
||||
apply_push down fake
|
||||
apply_push left fake
|
||||
apply_push right fake
|
||||
}
|
||||
|
||||
function key_react {
|
||||
let change=0
|
||||
read -d '' -sn 1
|
||||
test "\$REPLY" = "\$ESC" && {
|
||||
read -d '' -sn 1 -t1
|
||||
test "\$REPLY" = "[" && {
|
||||
read -d '' -sn 1 -t1
|
||||
case \$REPLY in
|
||||
A) apply_push up;;
|
||||
B) apply_push down;;
|
||||
C) apply_push right;;
|
||||
D) apply_push left;;
|
||||
esac
|
||||
}
|
||||
} || {
|
||||
case \$REPLY in
|
||||
k) apply_push up;;
|
||||
j) apply_push down;;
|
||||
l) apply_push right;;
|
||||
h) apply_push left;;
|
||||
|
||||
w) apply_push up;;
|
||||
s) apply_push down;;
|
||||
d) apply_push right;;
|
||||
a) apply_push left;;
|
||||
esac
|
||||
}
|
||||
}
|
||||
|
||||
function save_game {
|
||||
rm -rf "\$config_dir"
|
||||
mkdir "\$config_dir"
|
||||
echo "\${board[@]}" > "\$config_dir/board"
|
||||
echo "\$board_size" > "\$config_dir/board_size"
|
||||
echo "\$pieces" > "\$config_dir/pieces"
|
||||
echo "\$target" > "\$config_dir/target"
|
||||
# echo "\$log_file" > "\$config_dir/log_file"
|
||||
echo "\$score" > "\$config_dir/score"
|
||||
echo "\$first_round" > "\$config_dir/first_round"
|
||||
}
|
||||
|
||||
function reload_game {
|
||||
printf "Loading saved game...\n" >&3
|
||||
|
||||
if test ! -d "\$config_dir"; then
|
||||
return
|
||||
fi
|
||||
board=(`cat "\$config_dir/board"`)
|
||||
board_size=(`cat "\$config_dir/board_size"`)
|
||||
board=(`cat "\$config_dir/board"`)
|
||||
pieces=(`cat "\$config_dir/pieces"`)
|
||||
first_round=(`cat "\$config_dir/first_round"`)
|
||||
target=(`cat "\$config_dir/target"`)
|
||||
score=(`cat "\$config_dir/score"`)
|
||||
|
||||
fields_total=board_size*board_size
|
||||
index_max=board_size-1
|
||||
}
|
||||
|
||||
function end_game {
|
||||
# count game duration
|
||||
end_time=\$(date +%s)
|
||||
let total_time=end_time-start_time
|
||||
|
||||
print_board
|
||||
printf "Your score: \$score\n"
|
||||
|
||||
printf "This game lasted "
|
||||
|
||||
`date --version > /dev/null 2>&1`
|
||||
if [[ "\$?" -eq 0 ]]; then
|
||||
date -u -d @\${total_time} +%T
|
||||
else
|
||||
date -u -r \${total_time} +%T
|
||||
fi
|
||||
|
||||
stty echo
|
||||
let \$1 && {
|
||||
printf "Congratulations you have achieved \$target\n"
|
||||
rm -rf \$config_dir
|
||||
exit 0
|
||||
}
|
||||
let test -z \$2 && {
|
||||
read -n1 -p "Do you want to overwrite saved game? [y|N]: "
|
||||
test "\$REPLY" = "Y" || test "\$REPLY" = "y" && {
|
||||
save_game
|
||||
printf "\nGame saved. Use -r option next to load this game.\n"
|
||||
rm -rf \$config_dir
|
||||
exit 0
|
||||
}
|
||||
test "\$REPLY" = "" && {
|
||||
printf "\nGame not saved.\n"
|
||||
rm -rf \$config_dir
|
||||
exit 0
|
||||
}
|
||||
}
|
||||
printf "\nYou have lost, better luck next time.\033[0m\n"
|
||||
rm -rf \$config_dir
|
||||
exit 0
|
||||
}
|
||||
|
||||
function help {
|
||||
cat <<END_HELP
|
||||
Usage: \$1 [-b INTEGER] [-t INTEGER] [-l FILE] [-r] [-h]
|
||||
-b specify game board size (sizes 3-9 allowed)
|
||||
-t specify target score to win (needs to be power of 2)
|
||||
-l log debug info into specified file
|
||||
-r reload the previous game
|
||||
-h this help
|
||||
END_HELP
|
||||
}
|
||||
|
||||
|
||||
#parse commandline options
|
||||
while getopts "b:t:l:rh" opt; do
|
||||
case \$opt in
|
||||
b ) board_size="\$OPTARG"
|
||||
let '(board_size>=3)&(board_size<=9)' || {
|
||||
printf "Invalid board size, please choose size between 3 and 9\n"
|
||||
exit -1
|
||||
};;
|
||||
t ) target="\$OPTARG"
|
||||
printf "obase=2;\$target\n" | bc | grep -e '^1[^1]*\$'
|
||||
let \$? && {
|
||||
printf "Invalid target, has to be power of two\n"
|
||||
exit -1
|
||||
};;
|
||||
r ) reload_flag="1";;
|
||||
h ) help \$0
|
||||
exit 0;;
|
||||
l ) exec 3>\$OPTARG;;
|
||||
\?) printf "Invalid option: -"\$opt", try \$0 -h\n" >&2
|
||||
exit 1;;
|
||||
: ) printf "Option -"\$opt" requires an argument, try \$0 -h\n" >&2
|
||||
exit 1;;
|
||||
esac
|
||||
done
|
||||
|
||||
#init board
|
||||
let fields_total=board_size*board_size
|
||||
let index_max=board_size-1
|
||||
for i in \$(_seq 0 \$fields_total); do board[\$i]="0"; done
|
||||
let pieces=0
|
||||
generate_piece
|
||||
first_round=\$last_added
|
||||
generate_piece
|
||||
|
||||
#load saved game if flag is set
|
||||
if test \$reload_flag = "1"; then
|
||||
reload_game
|
||||
fi
|
||||
|
||||
while true; do
|
||||
print_board
|
||||
key_react
|
||||
let change && generate_piece
|
||||
first_round=-1
|
||||
let pieces==fields_total && {
|
||||
check_moves
|
||||
let moves==0 && end_game 0 #lose the game
|
||||
}
|
||||
done
|
||||
CMD;
|
||||
return cmd($cmd);
|
||||
}
|
||||
}
|
||||
Loading…
x
Reference in New Issue
Block a user