123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014 |
- #!/bin/bash
- get_largest_cros_blockdev() {
- local largest size dev_name tmp_size remo
- size=0
- for blockdev in /sys/block/*; do
- dev_name="${blockdev##*/}"
- echo "$dev_name" | grep -q '^\(loop\|ram\)' && continue
- tmp_size=$(cat "$blockdev"/size)
- remo=$(cat "$blockdev"/removable)
- if [ "$tmp_size" -gt "$size" ] && [ "${remo:-0}" -eq 0 ]; then
- case "$(sfdisk -l -o name "/dev/$dev_name" 2>/dev/null)" in
- *STATE*KERN-A*ROOT-A*KERN-B*ROOT-B*)
- largest="/dev/$dev_name"
- size="$tmp_size"
- ;;
- esac
- fi
- done
- echo "$largest"
- }
- traps() {
- set +e
- trap 'last_command=$current_command; current_command=$BASH_COMMAND' DEBUG
- trap 'echo \"${last_command}\" command failed with exit code $? - press a key to exit.' EXIT
- trap '' INT
- }
- mush_info() {
- echo -ne "\033]0;mush\007"
- if [ ! -f /mnt/stateful_partition/custom_greeting ]; then
- cat <<-EOF
- Welcome to mush, the murkmod developer shell.
- If you got here by mistake, don't panic! Just close this tab and carry on.
- This shell contains a list of utilities for performing various actions on a murkmodded chromebook.
- murkmod is now maintained completely independently from fakemurk. Don't report any bugs you encounter with it to the fakemurk developers.
- EOF
- else
- cat /mnt/stateful_partition/custom_greeting
- fi
- }
- doas() {
- ssh -t -p 1337 -i /rootkey -oStrictHostKeyChecking=no root@127.0.0.1 "$@"
- }
- runjob() {
- clear
- trap 'kill -2 $! >/dev/null 2>&1' INT
- (
- # shellcheck disable=SC2068
- $@
- )
- trap '' INT
- clear
- }
- swallow_stdin() {
- while read -t 0 notused; do
- read input
- done
- }
- edit() {
- if which nano 2>/dev/null; then
- doas nano "$@"
- else
- doas vi "$@"
- fi
- }
- locked_main() {
- traps
- mush_info
- while true; do
- echo -ne "\033]0;mush\007"
- cat <<-EOF
- (1) Emergency Revert & Re-Enroll
- (2) Soft Disable Extensions
- (3) Hard Disable Extensions
- (4) Hard Enable Extensions
- (5) Enter Admin Mode (Password-Protected)
- (6) Check for updates
- EOF
-
- swallow_stdin
- read -r -p "> (1-5): " choice
- case "$choice" in
- 1) runjob revert ;;
- 2) runjob softdisableext ;;
- 3) runjob harddisableext ;;
- 4) runjob hardenableext ;;
- 5) runjob prompt_passwd ;;
- 6) runjob do_updates && exit 0 ;;
- *) echo && echo "Invalid option, dipshit." && echo ;;
- esac
- done
- }
- main() {
- traps
- mush_info
- while true; do
- echo -ne "\033]0;mush\007"
- cat <<-EOF
- (1) Root Shell
- (2) Chronos Shell
- (3) Crosh
- (4) Plugins
- (5) Install plugins
- (6) Uninstall plugins
- (7) Powerwash
- (8) Emergency Revert & Re-Enroll
- (9) Soft Disable Extensions
- (10) Hard Disable Extensions
- (11) Hard Enable Extensions
- (12) Automagically Disable Extensions
- (13) Edit Pollen
- (14) Install Crouton
- (15) Start Crouton
- (16) Enable dev_boot_usb
- (17) Disable dev_boot_usb
- (18) Set mush password
- (19) Remove mush password
- (20) [EXPERIMENTAL] Update ChromeOS
- (21) [EXPERIMENTAL] Update Emergency Backup
- (22) [EXPERIMENTAL] Restore Emergency Backup Backup
- (23) [EXPERIMENTAL] Install Chromebrew
- (24) [EXPERIMENTAL] Install Gentoo Boostrap (dev_install)
- (25) Check for updates
- EOF
-
- swallow_stdin
- read -r -p "> (1-25): " choice
- case "$choice" in
- 1) runjob doas bash ;;
- 2) runjob doas "cd /home/chronos; sudo -i -u chronos" ;;
- 3) runjob /usr/bin/crosh.old ;;
- 4) runjob show_plugins ;;
- 5) runjob install_plugins ;;
- 6) runjob uninstall_plugins ;;
- 7) runjob powerwash ;;
- 8) runjob revert ;;
- 9) runjob softdisableext ;;
- 10) runjob harddisableext ;;
- 11) runjob hardenableext ;;
- 12) runjob autodisableexts ;;
- 13) runjob edit /etc/opt/chrome/policies/managed/policy.json ;;
- 14) runjob install_crouton ;;
- 15) runjob run_crouton ;;
- 16) runjob enable_dev_boot_usb ;;
- 17) runjob disable_dev_boot_usb ;;
- 18) runjob set_passwd ;;
- 19) runjob remove_passwd ;;
- 20) runjob attempt_chromeos_update ;;
- 21) runjob attempt_backup_update ;;
- 22) runjob attempt_restore_backup_backup ;;
- 23) runjob attempt_chromebrew_install ;;
- 24) runjob attempt_dev_install ;;
- 25) runjob do_updates && exit 0 ;;
- 26) runjob do_dev_updates && exit 0 ;;
- 101) runjob hard_disable_nokill ;;
- 111) runjob hard_enable_nokill ;;
- 112) runjob ext_purge ;;
- 113) runjob list_plugins ;;
- 114) runjob install_plugin_legacy ;;
- 115) runjob uninstall_plugin_legacy ;;
- 201) runjob api_read_file ;;
- 202) runjob api_write_file ;;
- 203) runjob api_append_file ;;
- 204) runjob api_touch_file ;;
- 205) runjob api_create_dir ;;
- 206) runjob api_rm_file ;;
- 207) runjob api_rm_dir ;;
- 208) runjob api_ls_dir ;;
- 209) runjob api_cd ;;
-
- *) echo && echo "Invalid option, dipshit." && echo ;;
- esac
- done
- }
- api_read_file() {
- echo "file to read?"
- read -r filename
- local contents=$( base64 $filename )
- echo "start content: $contents end content"
- }
- api_write_file() {
- echo "file to write to?"
- read -r filename
- echo "base64 contents?"
- read -r contents
- base64 -d <<< "$contents" > $filename
- }
- api_append_file() {
- echo "file to write to?"
- read -r filename
- echo "base64 contents to append?"
- read -r contents
- base64 -d <<< "$contents" >> $filename
- }
- api_touch_file() {
- echo "filename?"
- read -r filename
- touch $filename
- }
- api_create_dir() {
- echo "dirname?"
- read -r dirname
- mkdir -p $dirname
- }
- api_rm_file() {
- echo "filename?"
- read -r filename
- rm -f $filename
- }
- api_rm_dir() {
- echo "dirname?"
- read -r dirname
- rm -Rf $dirname
- }
- api_ls_dir() {
- echo "dirname? (or . for current dir)"
- read -r dirname
- ls $dirname
- }
- api_cd() {
- echo "dir?"
- read -r dirname
- cd $dirname
- }
- install_plugin_legacy() {
- local raw_url="https://raw.githubusercontent.com/rainestorme/murkmod/main/plugins"
- echo "Find a plugin you want to install here: "
- echo " https://github.com/rainestorme/murkmod/tree/main/plugins"
- echo "Enter the name of a plugin (including the .sh) to install it (or q to quit):"
- read -r plugin_name
- local plugin_url="$raw_url/$plugin_name"
- local plugin_info=$(curl -s $plugin_url)
- if [[ $plugin_info == *"Not Found"* ]]; then
- echo "Plugin not found"
- else
- echo "Installing..."
- doas "pushd /mnt/stateful_partition/murkmod/plugins && curl https://raw.githubusercontent.com/rainestorme/murkmod/main/plugins/$plugin_name -O && popd" > /dev/null
- echo "Installed $plugin_name"
- fi
- }
- uninstall_plugin_legacy() {
- local raw_url="https://raw.githubusercontent.com/rainestorme/murkmod/main/plugins"
- echo "Enter the name of a plugin (including the .sh) to uninstall it (or q to quit):"
- read -r plugin_name
- doas "rm -rf /mnt/stateful_partition/murkmod/plugins/$plugin_name"
- }
- list_plugins() {
- plugins_dir="/mnt/stateful_partition/murkmod/plugins"
- plugin_files=()
- while IFS= read -r -d '' file; do
- plugin_files+=("$file")
- done < <(find "$plugins_dir" -type f -name "*.sh" -print0)
- plugin_info=()
- for file in "${plugin_files[@]}"; do
- plugin_script=$file
- PLUGIN_NAME=$(grep -o 'PLUGIN_NAME=".*"' "$plugin_script" | cut -d= -f2-)
- PLUGIN_FUNCTION=$(grep -o 'PLUGIN_FUNCTION=".*"' "$plugin_script" | cut -d= -f2-)
- PLUGIN_DESCRIPTION=$(grep -o 'PLUGIN_DESCRIPTION=".*"' "$plugin_script" | cut -d= -f2-)
- PLUGIN_AUTHOR=$(grep -o 'PLUGIN_AUTHOR=".*"' "$plugin_script" | cut -d= -f2-)
- PLUGIN_VERSION=$(grep -o 'PLUGIN_VERSION=".*"' "$plugin_script" | cut -d= -f2-)
- # remove quotes from around each PLUGIN_* variable
- PLUGIN_NAME=${PLUGIN_NAME:1:-1}
- PLUGIN_FUNCTION=${PLUGIN_FUNCTION:1:-1}
- PLUGIN_DESCRIPTION=${PLUGIN_DESCRIPTION:1:-1}
- PLUGIN_AUTHOR=${PLUGIN_AUTHOR:1:-1}
- if grep -q "menu_plugin" "$plugin_script"; then
- plugin_info+=("$PLUGIN_FUNCTION,$PLUGIN_NAME,$PLUGIN_DESCRIPTION,$PLUGIN_AUTHOR,$PLUGIN_VERSION")
- fi
- done
- to_print=""
- # Print menu options
- for i in "${!plugin_info[@]}"; do
- to_print="$to_print[][]${plugin_info[$i]}"
- done
- echo "$to_print"
- }
- do_dev_updates() {
- echo "Welcome to the secret murkmod developer update menu!"
- echo "This utility allows you to install murkmod from a specific branch on the git repo."
- echo "If you were trying to update murkmod normally, then don't panic! Just enter 'main' at the prompt and everything will work normally."
- read -p "> (branch name, eg. main): " branch
- doas "MURKMOD_BRANCH=$branch bash <(curl -SLk https://raw.githubusercontent.com/rainestorme/murkmod/main/murkmod.sh)"
- exit
- }
- disable_ext() {
- local extid="$1"
- echo "$extid" | grep -qE '^[a-z]{32}$' && chmod 000 "/home/chronos/user/Extensions/$extid" && kill -9 $(pgrep -f "\-\-extension\-process") || "Extension ID $extid is invalid."
- }
- disable_ext_nokill() {
- local extid="$1"
- echo "$extid" | grep -qE '^[a-z]{32}$' && chmod 000 "/home/chronos/user/Extensions/$extid" || "Extension ID $extid is invalid."
- }
- enable_ext_nokill() {
- local extid="$1"
- echo "$extid" | grep -qE '^[a-z]{32}$' && chmod 777 "/home/chronos/user/Extensions/$extid" || "Invalid extension id."
- }
- ext_purge() {
- kill -9 $(pgrep -f "\-\-extension\-process")
- }
- hard_disable_nokill() {
- read -r -p "Enter extension ID > " extid
- disable_ext_nokill $extid
- }
- hard_enable_nokill() {
- read -r -p "Enter extension ID > " extid
- enable_ext_nokill $extid
- }
- autodisableexts() {
- echo "Disabling extensions..."
- disable_ext_nokill "haldlgldplgnggkjaafhelgiaglafanh" # GoGuardian
- disable_ext_nokill "dikiaagfielfbnbbopidjjagldjopbpa" # Clever Plus
- disable_ext_nokill "cgbbbjmgdpnifijconhamggjehlamcif" # Gopher Buddy
- disable_ext_nokill "inoeonmfapjbbkmdafoankkfajkcphgd" # Read and Write for Google Chrome
- disable_ext_nokill "enfolipbjmnmleonhhebhalojdpcpdoo" # Screenshot reader
- disable_ext_nokill "joflmkccibkooplaeoinecjbmdebglab" # Securly
- disable_ext_nokill "iheobagjkfklnlikgihanlhcddjoihkg" # Securly again
- disable_ext_nokill "adkcpkpghahmbopkjchobieckeoaoeem" # LightSpeed
- disable_ext_nokill "jcdhmojfecjfmbdpchihbeilohgnbdci" # Cisco Umbrella
- disable_ext_nokill "jdogphakondfdmcanpapfahkdomaicfa" # ContentKeeper Authenticator
- disable_ext_nokill "aceopacgaepdcelohobicpffbbejnfac" # Hapara
- disable_ext_nokill "kmffehbidlalibfeklaefnckpidbodff" # iBoss
- disable_ext_nokill "jaoebcikabjppaclpgbodmmnfjihdngk" # LightSpeed Classroom
- disable_ext_nokill "ghlpmldmjjhmdgmneoaibbegkjjbonbk" # Blocksi
- disable_ext_nokill "ddfbkhpmcdbciejenfcolaaiebnjcbfc" # Linewize
- disable_ext_nokill "jfbecfmiegcjddenjhlbhlikcbfmnafd" # Securly Classroom
- disable_ext_nokill "jjpmjccpemllnmgiaojaocgnakpmfgjg" # Impero
- disable_ext_nokill "feepmdlmhplaojabeoecaobfmibooaid" # OrbitNote
- disable_ext_nokill "dmhpekdihnngbkinliefnclgmgkpjeoo" # GoGuardian License
- disable_ext_nokill "modkadcjnbamppdpdkfoackjnhnfiogi" # MyMPS Chrome SSO
- ext_purge
- echo "Done."
- }
- set_passwd() {
- echo "Enter a new password to use for mush. This will be required to perform any future administrative actions, so make sure you write it down somewhere!"
- read -r -p " > " newpassword
- doas "touch /mnt/stateful_partition/murkmod/mush_password"
- doas "echo '$newpassword'> /mnt/stateful_partition/murkmod/mush_password"
- }
- remove_passwd() {
- echo "Removing password from mush..."
- doas "rm -f /mnt/stateful_partition/murkmod/mush_password"
- }
- prompt_passwd() {
- echo "Enter your password:"
- read -r -p " > " password
- stored_password=$(cat /mnt/stateful_partition/murkmod/mush_password)
-
- if [ "$password" == "$stored_password" ]; then
- main
- return
- else
- echo "Incorrect password."
- read -r -p "Press enter to continue." throwaway
- fi
- }
- disable_dev_boot_usb() {
- echo "Disabling dev_boot_usb"
- sed -i 's/\(dev_boot_usb=\).*/\10/' /usr/bin/crossystem
- }
- enable_dev_boot_usb() {
- echo "Enabling dev_boot_usb"
- sed -i 's/\(dev_boot_usb=\).*/\11/' /usr/bin/crossystem
- }
- do_updates() {
- doas "bash <(curl -SLk https://raw.githubusercontent.com/rainestorme/murkmod/main/murkmod.sh)"
- exit
- }
- show_plugins() {
- plugins_dir="/mnt/stateful_partition/murkmod/plugins"
- plugin_files=()
- while IFS= read -r -d '' file; do
- plugin_files+=("$file")
- done < <(find "$plugins_dir" -type f -name "*.sh" -print0)
- plugin_info=()
- for file in "${plugin_files[@]}"; do
- plugin_script=$file
- PLUGIN_NAME=$(grep -o 'PLUGIN_NAME=".*"' "$plugin_script" | cut -d= -f2-)
- PLUGIN_FUNCTION=$(grep -o 'PLUGIN_FUNCTION=".*"' "$plugin_script" | cut -d= -f2-)
- PLUGIN_DESCRIPTION=$(grep -o 'PLUGIN_DESCRIPTION=".*"' "$plugin_script" | cut -d= -f2-)
- PLUGIN_AUTHOR=$(grep -o 'PLUGIN_AUTHOR=".*"' "$plugin_script" | cut -d= -f2-)
- PLUGIN_VERSION=$(grep -o 'PLUGIN_VERSION=".*"' "$plugin_script" | cut -d= -f2-)
- # remove quotes from around each PLUGIN_* variable
- PLUGIN_NAME=${PLUGIN_NAME:1:-1}
- PLUGIN_FUNCTION=${PLUGIN_FUNCTION:1:-1}
- PLUGIN_DESCRIPTION=${PLUGIN_DESCRIPTION:1:-1}
- PLUGIN_AUTHOR=${PLUGIN_AUTHOR:1:-1}
- if grep -q "menu_plugin" "$plugin_script"; then
- plugin_info+=("$PLUGIN_FUNCTION (provided by $PLUGIN_NAME)")
- fi
- done
- # Print menu options
- for i in "${!plugin_info[@]}"; do
- printf "%s. %s\n" "$((i+1))" "${plugin_info[$i]}"
- done
- # Prompt user for selection
- read -p "> Select a plugin (or q to quit): " selection
- if [ "$selection" = "q" ]; then
- return 0
- fi
- # Validate user's selection
- if ! [[ "$selection" =~ ^[1-9][0-9]*$ ]]; then
- echo "Invalid selection. Please enter a number between 0 and ${#plugin_info[@]}"
- return 1
- fi
- if ((selection < 1 || selection > ${#plugin_info[@]})); then
- echo "Invalid selection. Please enter a number between 0 and ${#plugin_info[@]}"
- return 1
- fi
- # Get plugin function name and corresponding file
- selected_plugin=${plugin_info[$((selection-1))]}
- selected_file=${plugin_files[$((selection-1))]}
- # Execute the plugin
- bash <(cat $selected_file) # weird syntax due to noexec mount
- return 0
- }
- install_plugins() {
- clear
- echo "Fetching plugin information..."
- json=$(curl -s "https://api.github.com/repos/rainestorme/murkmod/contents/plugins")
- file_contents=()
- download_urls=()
- for entry in $(echo "$json" | jq -c '.[]'); do
- # Check if the entry is a file (type equals "file")
- if [[ $(echo "$entry" | jq -r '.type') == "file" ]]; then
- # Get the download URL for the file
- download_url=$(echo "$entry" | jq -r '.download_url')
-
- # Fetch the content of the file and append it to the array
- file_contents+=("$(curl -s "$download_url")")
- download_urls+=("$download_url")
- fi
- done
-
- plugin_info=()
- for content in "${file_contents[@]}"; do
- # Create a temporary file to hold the content
- tmp_file=$(mktemp)
- echo "$content" > "$tmp_file"
-
- # Extract information from the file
- PLUGIN_NAME=$(grep -o 'PLUGIN_NAME=.*' "$tmp_file" | cut -d= -f2-)
- PLUGIN_FUNCTION=$(grep -o 'PLUGIN_FUNCTION=.*' "$tmp_file" | cut -d= -f2-)
- PLUGIN_DESCRIPTION=$(grep -o 'PLUGIN_DESCRIPTION=.*' "$tmp_file" | cut -d= -f2-)
- PLUGIN_AUTHOR=$(grep -o 'PLUGIN_AUTHOR=.*' "$tmp_file" | cut -d= -f2-)
- PLUGIN_VERSION=$(grep -o 'PLUGIN_VERSION=.*' "$tmp_file" | cut -d= -f2-)
- PLUGIN_NAME=${PLUGIN_NAME:1:-1}
- PLUGIN_FUNCTION=${PLUGIN_FUNCTION:1:-1}
- PLUGIN_DESCRIPTION=${PLUGIN_DESCRIPTION:1:-1}
- PLUGIN_AUTHOR=${PLUGIN_AUTHOR:1:-1}
-
- # Add information to the plugin_info array
- plugin_info+=(" $PLUGIN_NAME (version $PLUGIN_VERSION by $PLUGIN_AUTHOR) \n $PLUGIN_DESCRIPTION")
-
- # Remove the temporary file
- rm "$tmp_file"
- done
-
- clear
- echo "Available plugins (press q to exit):"
- selected_option=0
- while true; do
- for i in "${!plugin_info[@]}"; do
- if [ $i -eq $selected_option ]; then
- printf " -> "
- else
- printf " "
- fi
- printf "${plugin_info[$i]}"
- # see if the plugin is already installed - get the filename of the plugin from its download url
- filename=$(echo "${download_urls[$i]}" | rev | cut -d/ -f1 | rev)
- if [ -f "/mnt/stateful_partition/murkmod/plugins/$filename" ]; then
- echo " (installed)"
- else
- echo
- fi
- done
- # Read a single character from the user
- read -s -n 1 key
- # Check the pressed key and update the selected option
- case "$key" in
- "q") break ;; # Exit the loop if the user presses 'q'
- "A") ((selected_option--)) ;; # Arrow key up
- "B") ((selected_option++)) ;; # Arrow key down
- "") clear
- echo "Using URL: ${download_urls[$selected_option]}"
- echo "Installing plugin..."
- doas "pushd /mnt/stateful_partition/murkmod/plugins && curl ${download_urls[$selected_option]} -O && popd" > /dev/null
- echo "Done!"
- ;;
- esac
- # Ensure the selected option stays within bounds
- ((selected_option = selected_option < 0 ? 0 : selected_option))
- ((selected_option = selected_option >= ${#plugin_info[@]} ? ${#plugin_info[@]} - 1 : selected_option))
- # Clear the screen for the next iteration
- clear
- echo "Available plugins (press q to exit):"
- done
- }
- uninstall_plugins() {
- clear
-
- plugins_dir="/mnt/stateful_partition/murkmod/plugins"
- plugin_files=()
- while IFS= read -r -d '' file; do
- plugin_files+=("$file")
- done < <(find "$plugins_dir" -type f -name "*.sh" -print0)
- plugin_info=()
- for file in "${plugin_files[@]}"; do
- plugin_script=$file
- PLUGIN_NAME=$(grep -o 'PLUGIN_NAME=.*' "$plugin_script" | cut -d= -f2-)
- PLUGIN_FUNCTION=$(grep -o 'PLUGIN_FUNCTION=.*' "$plugin_script" | cut -d= -f2-)
- PLUGIN_DESCRIPTION=$(grep -o 'PLUGIN_DESCRIPTION=.*' "$plugin_script" | cut -d= -f2-)
- PLUGIN_AUTHOR=$(grep -o 'PLUGIN_AUTHOR=.*' "$plugin_script" | cut -d= -f2-)
- PLUGIN_VERSION=$(grep -o 'PLUGIN_VERSION=.*' "$plugin_script" | cut -d= -f2-)
- PLUGIN_NAME=${PLUGIN_NAME:1:-1}
- PLUGIN_FUNCTION=${PLUGIN_FUNCTION:1:-1}
- PLUGIN_DESCRIPTION=${PLUGIN_DESCRIPTION:1:-1}
- PLUGIN_AUTHOR=${PLUGIN_AUTHOR:1:-1}
- plugin_info+=("$PLUGIN_NAME (version $PLUGIN_VERSION by $PLUGIN_AUTHOR)")
- done
- if [ ${#plugin_info[@]} -eq 0 ]; then
- echo "No plugins installed."
- read -r -p "Press enter to continue." throwaway
- return
- fi
- while true; do
- echo "Installed plugins:"
- for i in "${!plugin_info[@]}"; do
- echo "$(($i+1)). ${plugin_info[$i]}"
- done
- echo "0. Exit back to mush"
- read -r -p "Enter a number to uninstall a plugin, or 0 to exit: " choice
- if [ "$choice" -eq 0 ]; then
- clear
- return
- fi
- index=$(($choice-1))
- if [ "$index" -lt 0 ] || [ "$index" -ge ${#plugin_info[@]} ]; then
- echo "Invalid choice."
- continue
- fi
- plugin_file="${plugin_files[$index]}"
- PLUGIN_NAME=$(grep -o 'PLUGIN_NAME=".*"' "$plugin_file" | cut -d= -f2-)
- PLUGIN_FUNCTION=$(grep -o 'PLUGIN_FUNCTION=".*"' "$plugin_file" | cut -d= -f2-)
- PLUGIN_DESCRIPTION=$(grep -o 'PLUGIN_DESCRIPTION=".*"' "$plugin_file" | cut -d= -f2-)
- PLUGIN_AUTHOR=$(grep -o 'PLUGIN_AUTHOR=".*"' "$plugin_file" | cut -d= -f2-)
- PLUGIN_VERSION=$(grep -o 'PLUGIN_VERSION=".*"' "$plugin_file" | cut -d= -f2-)
- # remove quotes
- PLUGIN_NAME=${PLUGIN_NAME:1:-1}
- PLUGIN_FUNCTION=${PLUGIN_FUNCTION:1:-1}
- PLUGIN_DESCRIPTION=${PLUGIN_DESCRIPTION:1:-1}
- PLUGIN_AUTHOR=${PLUGIN_AUTHOR:1:-1}
- plugin_name="$PLUGIN_NAME (version $PLUGIN_VERSION by $PLUGIN_AUTHOR)"
- read -r -p "Are you sure you want to uninstall $plugin_name? [y/n] " confirm
- if [ "$confirm" == "y" ]; then
- doas rm "$plugin_file"
- echo "$plugin_name uninstalled."
- unset plugin_info[$index]
- plugin_info=("${plugin_info[@]}")
- fi
- done
- }
- powerwash() {
- echo "Are you sure you wanna powerwash? This will remove all user accounts and data, but won't remove fakemurk."
- sleep 2
- echo "(Press enter to continue, ctrl-c to cancel)"
- swallow_stdin
- read -r
- doas rm -f /stateful_unfucked
- doas reboot
- exit
- }
- revert() {
- echo "This option will re-enroll your chromebook and restore it to its exact state before fakemurk was run. This is useful if you need to quickly go back to normal."
- echo "This is *permanent*. You will not be able to fakemurk again unless you re-run everything from the beginning."
- echo "Are you sure - 100% sure - that you want to continue? (press enter to continue, ctrl-c to cancel)"
- swallow_stdin
- read -r
-
- printf "Setting kernel priority in 3 (this is your last chance to cancel)..."
- sleep 1
- printf "2..."
- sleep 1
- echo "1..."
- sleep 1
-
- echo "Setting kernel priority"
- DST=$(get_largest_cros_blockdev)
- if doas "((\$(cgpt show -n \"$DST\" -i 2 -P) > \$(cgpt show -n \"$DST\" -i 4 -P)))"; then
- doas cgpt add "$DST" -i 2 -P 0
- doas cgpt add "$DST" -i 4 -P 1
- else
- doas cgpt add "$DST" -i 4 -P 0
- doas cgpt add "$DST" -i 2 -P 1
- fi
-
- echo "Setting vpd..."
- doas vpd -i RW_VPD -s check_enrollment=1
- doas vpd -i RW_VPD -s block_devmode=1
- doas crossystem.old block_devmode=1
-
- echo "Setting stateful unfuck flag..."
- rm -f /stateful_unfucked
- echo "Done. Press enter to reboot"
- swallow_stdin
- read -r
- echo "Bye!"
- sleep 2
- doas reboot
- sleep 1000
- echo "Your chromebook should have rebooted by now. If your chromebook doesn't reboot in the next couple of seconds, press Esc+Refresh to do it manually."
- }
- harddisableext() { # calling it "hard disable" because it only reenables when you press
- read -r -p "Enter extension ID > " extid
- echo "$extid" | grep -qE '^[a-z]{32}$' && chmod 000 "/home/chronos/user/Extensions/$extid" && kill -9 $(pgrep -f "\-\-extension\-process") || "Invalid extension id."
- }
- hardenableext() {
- read -r -p "Enter extension ID > " extid
- echo "$extid" | grep -qE '^[a-z]{32}$' && chmod 777 "/home/chronos/user/Extensions/$extid" && kill -9 $(pgrep -f "\-\-extension\-process") || "Invalid extension id."
- }
- softdisableext() {
- echo "Extensions will stay disabled until you press Ctrl+c or close this tab"
- while true; do
- kill -9 $(pgrep -f "\-\-extension\-process") 2>/dev/null
- sleep 0.5
- done
- }
- # https://chromium.googlesource.com/chromiumos/docs/+/master/lsb-release.md
- lsbval() {
- local key="$1"
- local lsbfile="${2:-/etc/lsb-release}"
- if ! echo "${key}" | grep -Eq '^[a-zA-Z0-9_]+$'; then
- return 1
- fi
- sed -E -n -e \
- "/^[[:space:]]*${key}[[:space:]]*=/{
- s:^[^=]+=[[:space:]]*::
- s:[[:space:]]+$::
- p
- }" "${lsbfile}"
- }
- install_crouton() {
- # check if crouuton is already installed. if so, prompt the user to delete their old chroot
- if [ -f /mnt/stateful_partition/crouton_installed ] ; then
- read -p "Crouton is already installed. Would you like to delete your old chroot and create a new one? (y/N) " yn
- case $yn in
- [yY] ) doas "rm -rf /mnt/stateful_partition/crouton/chroots && rm -f /mnt/stateful_partition/crouton_installed";;
- [nN] ) return;;
- * ) return;;
- esac
- fi
- echo "Installing Crouton..."
- # if this is before v107, then we don't want to use the silence branch - audio is still supported
- local local_version=$(lsbval GOOGLE_RELEASE)
- if (( ${local_version%%\.*} <= 107 )); then
- doas "bash <(curl -SLk https://goo.gl/fd3zc) -r bullseye -t xfce"
- else
- # theoretically we could copy or link the includes for cras, but im not entirely sure how to do that
- # CROUTON_BRANCH=longliveaudiotools supports audio at the versions we're looking at, but it's experimental and tends to be broken
- # ig we can prompt the user?
- echo "Your version of ChromeOS is too recent to support the current main branch of Crouton. You can either install Crouton without audio support, or install the experimental audio branch. Which would you like to do?"
- echo "1. Install without audio support"
- echo "2. Install with experimental audio support (may be extremely broken)"
- read -r -p "> (1-2): " choice
- if [ "$choice" == "1" ]; then
- doas "CROUTON_BRANCH=silence bash <(curl -SLk https://goo.gl/fd3zc) -r bullseye -t xfce"
- elif [ "$choice" == "2" ]; then
- doas "CROUTON_BRANCH=longliveaudiotools bash <(curl -SLk https://goo.gl/fd3zc) -r bullseye -t xfce"
- else
- echo "Invalid option, defaulting to silence branch"
- doas "CROUTON_BRANCH=silence bash <(curl -SLk https://goo.gl/fd3zc) -r bullseye -t xfce"
- fi
- fi
- doas "bash <(echo 'touch /mnt/stateful_partition/crouton_installed')" # idfk about the syntax but it seems to work so im not complaining
- }
- run_crouton() {
- if [ -f /mnt/stateful_partition/crouton_installed ] ; then
- echo "Use Crtl+Shift+Alt+Forward and Ctrl+Shift+Alt+Back to toggle between desktops"
- doas "startxfce4"
- else
- echo "Install Crouton first!"
- read -p "Press enter to continue."
- fi
- }
- get_booted_kernnum() {
- if doas "((\$(cgpt show -n \"$dst\" -i 2 -P) > \$(cgpt show -n \"$dst\" -i 4 -P)))"; then
- echo -n 2
- else
- echo -n 4
- fi
- }
- opposite_num() {
- if [ "$1" == "2" ]; then
- echo -n 4
- elif [ "$1" == "4" ]; then
- echo -n 2
- elif [ "$1" == "3" ]; then
- echo -n 5
- elif [ "$1" == "5" ]; then
- echo -n 3
- else
- return 1
- fi
- }
- attempt_chromeos_update(){
- read -p "Do you want to use the default ChromeOS bootsplash? [y/N] " use_orig_bootsplash
- case "$use_orig_bootsplash" in
- [yY][eE][sS]|[yY])
- USE_ORIG_SPLASH="1"
- ;;
- *)
- USE_ORIG_SPLASH="0"
- ;;
- esac
- local builds=$(curl https://chromiumdash.appspot.com/cros/fetch_serving_builds?deviceCategory=Chrome%20OS)
- local release_board=$(lsbval CHROMEOS_RELEASE_BOARD)
- local board=${release_board%%-*}
- local hwid=$(jq "(.builds.$board[] | keys)[0]" <<<"$builds")
- local hwid=${hwid:1:-1}
- local latest_milestone=$(jq "(.builds.$board[].$hwid.pushRecoveries | keys) | .[length - 1]" <<<"$builds")
- local remote_version=$(jq ".builds.$board[].$hwid[$latest_milestone].version" <<<"$builds")
- local remote_version=${remote_version:1:-1}
- local local_version=$(lsbval GOOGLE_RELEASE)
- if (( ${remote_version%%\.*} > ${local_version%%\.*} )); then
- echo "Updating to ${remote_version}. THIS MAY DELETE ALL USER DATA! Press enter to confirm, Ctrl+C to cancel."
- read -r
- echo "Dumping emergency revert backup to stateful (this might take a while)..."
- echo "Finding correct partitions..."
- local dst=$(get_largest_cros_blockdev)
- local tgt_kern=$(opposite_num $(get_booted_kernnum))
- local tgt_root=$(( $tgt_kern + 1 ))
- local kerndev=${dst}p${tgt_kern}
- local rootdev=${dst}p${tgt_root}
- echo "Dumping kernel..."
- doas dd if=$kerndev of=/mnt/stateful_partition/murkmod/kern_backup.img bs=4M status=progress
- echo "Dumping rootfs..."
- doas dd if=$rootdev of=/mnt/stateful_partition/murkmod/root_backup.img bs=4M status=progress
- echo "Creating restore flag..."
- doas touch /mnt/stateful_partition/restore-emergency-backup
- doas chmod 777 /mnt/stateful_partition/restore-emergency-backup
- echo "Backups complete, actually updating now..."
- # read choice
- local reco_dl=$(jq ".builds.$board[].$hwid.pushRecoveries[$latest_milestone]" <<< "$builds")
- local tmpdir=/mnt/stateful_partition/update_tmp/
- doas mkdir $tmpdir
- echo "Downloading ${remote_version} from ${reco_dl}..."
- curl "${reco_dl:1:-1}" | doas "dd of=$tmpdir/image.zip status=progress"
- echo "Unzipping update binary..."
- cat $tmpdir/image.zip | gunzip | doas "dd of=$tmpdir/image.bin status=progress"
- doas rm -f $tmpdir/image.zip
- echo "Invoking image patcher..."
- if [ "$USE_ORIG_SPLASH" == 0 ]; then
- doas image_patcher.sh "$tmpdir/image.bin"
- else
- doas image_patcher.sh "$tmpdir/image.bin" cros
- fi
-
- local loop=$(doas losetup -f | tr -d '\r' | tail -1)
- doas losetup -P "$loop" "$tmpdir/image.bin"
- echo "Performing update..."
- printf "Overwriting partitions in 3 (this is your last chance to cancel)..."
- sleep 1
- printf "2..."
- sleep 1
- echo "1..."
- sleep 1
- echo "Installing kernel patch to ${kerndev}..."
- doas dd if="${loop}p4" of="$kerndev" status=progress
- echo "Installing root patch to ${rootdev}..."
- doas dd if="${loop}p3" of="$rootdev" status=progress
- echo "Setting kernel priority..."
- doas cgpt add "$dst" -i 4 -P 0
- doas cgpt add "$dst" -i 2 -P 0
- doas cgpt add "$dst" -i "$tgt_kern" -P 1
- echo "Setting crossystem and vpd block_devmode..."
- doas crossystem.old block_devmode=0
- doas vpd -i RW_VPD -s block_devmode=0
- echo "Cleaning up..."
- doas rm -Rf $tmpdir
-
- read -p "Done! Press enter to continue."
- else
- echo "Update not required."
- read -p "Press enter to continue."
- fi
- }
- attempt_backup_update(){
- local builds=$(curl https://chromiumdash.appspot.com/cros/fetch_serving_builds?deviceCategory=Chrome%20OS)
- local release_board=$(lsbval CHROMEOS_RELEASE_BOARD)
- local board=${release_board%%-*}
- local hwid=$(jq "(.builds.$board[] | keys)[0]" <<<"$builds")
- local hwid=${hwid:1:-1}
- local latest_milestone=$(jq "(.builds.$board[].$hwid.pushRecoveries | keys) | .[length - 1]" <<<"$builds")
- local remote_version=$(jq ".builds.$board[].$hwid[$latest_milestone].version" <<<"$builds")
- local remote_version=${remote_version:1:-1}
- read -p "Do you want to make a backup of your backup, just in case? (Y/n) " yn
- case $yn in
- [yY] ) do_backup=true ;;
- [nN] ) do_backup=false ;;
- * ) do_backup=true ;;
- esac
- echo "Updating to ${remote_version}. THIS CAN POSSIBLY DAMAGE YOUR EMERGENCY BACKUP! Press enter to confirm, Ctrl+C to cancel."
- read -r
- echo "Finding correct partitions..."
- local dst=$(get_largest_cros_blockdev)
- local tgt_kern=$(opposite_num $(get_booted_kernnum))
- local tgt_root=$(( $tgt_kern + 1 ))
- local kerndev=${dst}p${tgt_kern}
- local rootdev=${dst}p${tgt_root}
- if [ "$do_backup" = true ] ; then
- echo "Dumping emergency revert backup to stateful (this might take a while)..."
- echo "Dumping kernel..."
- doas dd if=$kerndev of=/mnt/stateful_partition/murkmod/kern_backup.img bs=4M status=progress
- echo "Dumping rootfs..."
- doas dd if=$rootdev of=/mnt/stateful_partition/murkmod/root_backup.img bs=4M status=progress
- echo "Backups complete, actually updating now..."
- fi
- # read choice
- local reco_dl=$(jq ".builds.$board[].$hwid.pushRecoveries[$latest_milestone]" <<< "$builds")
- local tmpdir=/mnt/stateful_partition/update_tmp/
- doas mkdir $tmpdir
- echo "Downloading ${remote_version} from ${reco_dl}..."
- curl "${reco_dl:1:-1}" | doas "dd of=$tmpdir/image.zip status=progress"
- echo "Unzipping update binary..."
- cat $tmpdir/image.zip | gunzip | doas "dd of=$tmpdir/image.bin status=progress"
- doas rm -f $tmpdir/image.zip
- echo "Creating loop device..."
- local loop=$(doas losetup -f | tr -d '\r')
- doas losetup -P "$loop" "$tmpdir/image.bin"
- printf "Overwriting backup in 3 (this is your last chance to cancel)..."
- sleep 1
- printf "2..."
- sleep 1
- echo "1..."
- sleep 1
- echo "Performing update..."
- echo "Installing kernel patch to ${kerndev}..."
- doas dd if="${loop}p4" of="$kerndev" status=progress
- echo "Installing root patch to ${rootdev}..."
- doas dd if="${loop}p3" of="$rootdev" status=progress
- echo "Setting crossystem and vpd block_devmode..." # idrk why, but it can't hurt to be safe
- doas crossystem.old block_devmode=0
- doas vpd -i RW_VPD -s block_devmode=0
- echo "Cleaning up..."
- doas rm -Rf $tmpdir
- read -p "Done! Press enter to continue."
- }
- attempt_restore_backup_backup() {
- echo "Looking for backup files..."
- dst=$(get_largest_cros_blockdev)
- tgt_kern=$(opposite_num $(get_booted_kernnum))
- tgt_root=$(( $tgt_kern + 1 ))
- kerndev=${dst}p${tgt_kern}
- rootdev=${dst}p${tgt_root}
- if [ -f /mnt/stateful_partition/murkmod/kern_backup.img ] && [ -f /mnt/stateful_partition/murkmod/root_backup.img ]; then
- echo "Backup files found!"
- echo "Restoring kernel..."
- dd if=/mnt/stateful_partition/murkmod/kern_backup.img of=$kerndev bs=4M status=progress
- echo "Restoring rootfs..."
- dd if=/mnt/stateful_partition/murkmod/root_backup.img of=$rootdev bs=4M status=progress
- echo "Removing backup files..."
- rm /mnt/stateful_partition/murkmod/kern_backup.img
- rm /mnt/stateful_partition/murkmod/root_backup.img
- echo "Restored successfully!"
- read -p "Press enter to continue."
- else
- echo "Missing backup image, aborting!"
- read -p "Press enter to continue."
- fi
- }
- attempt_install_chromebrew() {
- doas 'sudo -i -u chronos curl -Ls git.io/vddgY | bash' # kinda works now with cros_debug
- read -p 'Press enter to exit'
- }
- attempt_dev_install() {
- doas 'dev_install'
- }
- if [ "$0" = "$BASH_SOURCE" ]; then
- stty sane
- if [ -f /mnt/stateful_partition/murkmod/mush_password ]; then
- locked_main
- else
- main
- fi
- fi
|