Mac terminal
Terminal Primer
Date back to the days when computing started with big mainframe computers, many users had to share access. They did this by typing and reading on a ‘terminal’ or ‘console’ — a dedicated device talking to the mainframe. Terminals used mechanical printers or typewriters to show the results, these were called teletypewriters or tty. The protocol that the mainframe used to communicate with the typewriters was named tty
as well. The protocol and its name have remained, long after the mechanical typewriters are gone. A tty
is a particular kind of device file that represents a terminal. In its most common meaning, terminal is synonymous with tty
. ttys001
it’s essentially “terminal input device #1”.
The Terminal application on macOS is a virtual terminal program to run a shell, either locally or to connect to servers. The terminal only provides a means to enter and display text. It will display a ‘prompt’ which tells the user, that the system is ready and the user can enter a command.
- A terminal is different than a shell. It used to be a hardware device, but now we use terminal emulators for similar functionality.
- Bash is just one of many shells, it happens to be the default shell in most Linux systems, but you may also be familiar with zsh, the default on macOS, and PowerShell or cmd.exe, the default on Windows.
A shell protects the user from the dangerous, complicated parts of the system and abstracts differences from one system to the other. Another way of looking at it, is that a shell protects the vulnerable, fragile parts of the system from the user. Technically, GUI which display files, folders and programs as icons in windows such as macOS, Windows, and even iOS and Android are also shells, since they shield the system internals from the user and vice versa. However, usually the term ‘shell’ is used for interfaces where the user has to type commands, or CLI. Command Line shells commonly have two major roles. The first is to interpret and execute commands entered in an interactive prompt and deliver the results back to the user. The second role is to interpret and process list of commands called scripts.
Many Unix and Unix-like systems have sh
as well as bash and other shells available. sh
goes back to the very early UNIX shells. sh
has survived because it serves as the lowest common standard for shell scripting. These standards are defined in the POSIX specification. When you have to build scripts that need to run across many different flavors and versions of Unix where you cannot rely on bash being present, then conforming to sh
might be necessary. (bash is also POSIX compliant, but it has more features). As a macOS administrator you should always choose /bin/bash
over /bin/sh
. You can check all the available shells in macOS cat /etc/shells
, show the current shell being used echo $SHELL
, and use chsh -s /bin/zsh
to change the current interactive shell.
POSIX (Portable Operating System Interface) is a family of standards, specified by the IEEE, to clarify and make uniform the application programming interfaces provided by Unix operating systems. When you write your programs to rely on POSIX standards, you can be pretty sure to be able to port them easily among a large family of Unix derivatives.
Finding Commands
All commands are files in the file system. They have a special file setting which makes them executable, so the system can interpret it as commands. If you want to know where a given command resides, you can use the which
command. When you enter a command without a path such as ls
, bash will start looking for the command executable in /usr/local/bin
, then in /usr/bin
, and then in /bin
(defined in PATH
). When it finds an executable ls
, stop looking and execute that.
Terminal–Finder Interaction
The CLI and the UI are not entirely separate areas in macOS, there is a lot of overlap and there are functions in Finder and Terminal that allow for quick interaction between them.
If you drag any folder from Finder to the Terminal application icon in the dock, Terminal will open a new window and change the working directory to the folder you dragged. When you drag any file into an open Terminal window, it will insert the full path to that file with spaces and other special characters properly escaped. If you prefer, you can get the same effect with copy and paste.
In Terminal if you type open .
, it will open the current working directory in a Finder. The open
command can do much more. In general you can think of open
as the command line equivalent of double-clicking a file or folder in Finder. open document.pdf
will open the PDF with the default application usually Preview. open https://google.com
will open the default browser, but you can use the -a
option to override the default application open -a Firefox https://google.com
.
Hidden Files
In UNIX, files or directories with a name beginning with .
are considered hidden and will not be shown in a normal file list with ls
. However you can list them with the option ls -a
. Usually dot files are configuration files or folders. macOS Sierra added a Finder keyboard shortcut to quickly show hidden files using Command + Shift + .
, which will quickly show all hidden files and a second time will re-hide them.
How to hide a file or folder in macOS Finder? Open the terminal and run chflags hidden <path>
. In case you need to set it back to visible, use chflags nohidden <path>
.
Deleting Files
In the Finder, deleted files are moved to the Trash, which is actually the invisible directory ~/.Trash
. There the file will remain until the user chooses ‘Empty Trash’. Only then is the file removed from disk. The command line has no such safety net. When you delete a file with the rm
command it is gone. You can add the -i
option to the rm
command which will ask for confirmation before actually deleting it. There is a command rmdir
which is the destructive equivalent of mkdir
. However, rmdir
can only remove empty directories.
The Clipboard
There are two commands specific to macOS that connect the clipboard closer to the shell commands. pbcopy
will take the contents of stdin
and put them in the clipboard. So anything you pipe into pbcopy
will end up in the clipboard, so you can paste it into a different place. For example, cat test.txt | pbcopy
is easier than open, select all, and copy. pbpaste
is the counterpart to pbcopy
. You can easily make the clipboard contents visible by typing pbpaste
as the next command.
>
sends stream to a file (overwrite) and>>
appends stream to a file, e.g., ls > a.txt
echo "" > a.txt
will still give you a file with one character (a newline).
cat /dev/null > a.txt
should clean all contents of a file.> /dev/null
means redirect the output to/dev/null
, which is a special file that discards all data written to it (essentially throwing away the output).
The ’[’ Marks
Using Terminal you may notice that there is a small gray square brackets before the prompt. They are called ‘Marks’ and every command that is executed automatically gets marked. You can quickly scroll the Terminal output to previous marks with Cmd + Up Arrow
and to the next mark with Cmd + Down Arrow
. You can hide them with ‘Hide Marks’ from the ‘View’ menu.
Navigating the Terminal Prompt
Instead of hitting the up arrow several times, you can also use Ctrl + R
and start typing a command you used before. This will search through the history backwards and recall the latest command you used starting with what you typed. history
command will print entire bash history to the screen (e.g. history | tail -5
shows the last 5 entries). By typing sudo !!
, the shell will substitute the !!
with the previous command, print the entire command after the substitution and immediately execute it with sudo
.
Once you have recalled a command and want to edit it, you will have to move the cursor. You can use Option + Left/Right Arrow
to move word by word. You can use Ctrl + A
to jump to the beginning of the line and Ctrl + E
to jump to the end. Ctrl + U
will clear the entire current line. Ctrl + W
delete the word before the cursor. And you can option-click with the mouse pointer on a character in the command line to move the cursor there.
Readline is a GNU library that gets a line from a user with editing. The bash shell is probably the most famous readline user (when you use
Ctrl+R
to search your history in bash, that feature actually comes from readline), but there are TONS of programs that use it – for example psql, irb, python3, etc.
Escaping Characters
The escape character in bash is the backslash \
. A character that follows a backslash will be treated with no special meaning. For a directory named Project (Old & New)
, you would type cd Project\ \(Old\ \&\ New\)
. Since escaping characters can make the path quite unreadable, you can also place the name in single quotes cd 'Project (Old & New)'
. Any character in single quotes is used as is, with no special function. (you cannot use single quotes when the filename contains a single quote). Tab-completion will escape spaces and other nasty characters automatically.
Quoting
- Quoting is used to remove special meanings from characters or words.
- single quotes - when the single quotes are used, every character within the quotes is preserved and is not evaluated.
- double quotes - when the double qoutes are used, the dollar sign, back quotes and blackslashes are evaluated and interpreted.
- escape character -
\
is used to preserve the literal value of the following character.
echo $HOME /home/user1/
echo \$HOME $HOME
echo '$HOME' $HOME
echo "$HOME" /home/user1/
Making tab-completion case-insensitive
The problem is that the file system of macOS is “case preserving, but case-insensitive”. That means the file system will remember wether you named a file README.TXT
, ReadMe.txt
or readme.TXT
and preserve that case, but using either of these will point to the same file. This may be confusing in Terminal. Since most other Unix file system are case-sensitive (i.e. README.TXT
and readme.txt
are different files) and most shells are case-sensitive too.
Don’t mix uppercase and lowercase letters in your filenames. It’s a recipe for disaster with different operating systems (some ignore the case while others don’t). I use kebab-case.
One thing you can change is wether tab-completion is case-sensitive or not. Since the underlying file system is insensitive, there is no reason tab-completion should be. To make tab-completion in bash case-insensitive, put set completion-ignore-case on
in your .inputrc
(create if necessary).
Viewing man
Pages
open x-man-page://ls
will open the man
page in a new yellow Terminal window, so you can still see what you are actually doing while reading the man page. Since this window shows the entire man page, you can scroll and use Command-F in this window. This behavior can also be achieved by right clicking on a word in a Terminal window and choose ‘Open man Page’ from the context menu.
For the normal man
page, this special display mode is actually controlled by the command less
. You can use /word<return>
to search in document, n
to find next occurrence of search term, N
to find previous occurrence of search term. (e.g. man ascii
quick access to ascii table, man man
lists the sections of the manual pages)
cat
show the contents of file, which is used for relatively small files.head
show the first part of the file.head -n 3 <filename>
prints first three lines.tail
show the last part of the file.tail -f /some/log/file
show the text appended to the file as the file grows.less
show contents of file one screen at a time. (more
is an old utility, just useless
and forget aboutmore
)
grep
Searches for pattern in files and prints each line that matches the input pattern grep -<options> <pattern> <filenames>
# options
-i ignore case
-n display line numbers along with lines
-c count the number of matching lines
# regular expression
[abc] matches any one of the characters in the square brackets
[0-9] matches any one of the characters in the range specified in the square brackets
^start matches the pattern only if the pattern is at the start of the line
end$ matches the pattern only if the pattern is at the end of the line
[^abc] matches any one character that is NOT present in the square brackets
. matches any one character
.* matches zero or more of any character
For example, ls -la | grep test | sort | uniq | wc -l
: The final output of this command would be the number of files and directories in the current directory whose name contains the string “test.”
When it comes to command line text processing, the three major pillars are
grep
for filtering,sed
for substitution andawk
for field processing. https://www-users.york.ac.uk/~mijp1/teaching/2nd_year_Comp_Lab/guides/grep_awk_sed.pdf
lsof
Linux/Unix considers everything as a file and lsof
is a command meaning “list open files”, which is used to report a list of all open files and the processes that opened them. Add -i
to list network connections. Use lsof -i -n -P | grep LISTEN
to check the listening ports and lsof -i :22
to see a specific port.
kill
command sends a kill signal to terminate any process gracefully when attached with a pid or a processname. This is the default and safest way to kill/terminate a or set of processes.kill <pid> / <processname>
sends SIGTERM (15) — Termination signal. However, this signal can be handled, ignored or caught in code. If the signal is not caught by a process, the process is killed.
kill -9
command sends a kill signal to terminate any process immediately when attached with a PID or a processname. It is a forceful way to kill/terminate a or set of processes.kill -9 <pid> / <processname>
sends SIGKILL (9) — Kill signal. This signal cannot be handled (caught), ignored or blocked. Hence, this immediately kill the process without any handling and this can create zombies process.
bash_profile and bashrc
There are two user level files which bash
may run when a bash shell starts. ~/.bash_profile
and ~/.bashrc
. The usual convention is that .bash_profile
will be executed at login shells, i.e. when you ssh into a remote host, it will ask you for user name and password to log in, so it is a login shell. But when you open a terminal, it does not ask for login and you will just get a command prompt. In other versions of Unix or Linux, this will not run the .bash_profile
but .bashrc
. The underlying idea is that the .bash_profile
should be run only when you login, and the .bashrc
for every new interactive shell. However, Terminal on macOS does not follow this convention. When Terminal opens a new window, it will run .bash_profile
.
- When you are living mostly on macOS and the Terminal, you can create a
.bash_profile
, ignore all the special cases and be happy. - If you want to have an approach that is more resilient to other terminal applications and might work across Unix/Linux platforms, put your configuration in
.bashrc
and source.bashrc
from.bash_profile
. - Usually the contents of a
.bash_profile
or.bashrc
will be setting environment variables, aliases, and functions.
There are more files which may be executed when a shell is created. When bash cannot find ~/.bash_profile
, it will look for ~/.bash_login
. When neither ~/.bash_profile
nor ~/.bash_login
exist, then ~/.profile
. If ~/.bash_profile
is present, the succeeding files will be ignored though you can source them in the .bash_profile
.
Environment Variables
The PATH
environment variable contains a list of directories that bash will search through for commands. echo $PATH
will show the current PATH
variable. If you want to know which command is actually executed, you can use the which
command to get the effective path like which python
or which bash
. You can add your own directories by either appending (safe) or prepending (risky) your own directories to the PATH
variable such as export PATH=$PATH:~/bin
. Since bash stops looking when it finds a match, the order of the directories in the PATH
is important. When you place your directories before the default directories you can override some of the system commands. This can be dangerous. You should add your directory to the end of the PATH
to be safe.
Changing the Editor
Some terminal commands will open a text editor directly. For example git
may open a text editor so you can enter a commit message. By default the system will open vi
but you can change the editor that will be used. The environment variable for this behavior is EDITOR
. So you can set the editor using export EDITOR=vim
or export EDITOR=nano
or export EDITOR=emacs
. (The export
command tells bash it should export this variable to child processes.)
You can also tell bash to color for some commands such as ls
. To do that you just need to set export CLICOLOR=1
and use unset CLICOLOR
to remove that variable.
Configuring aliases
Bash aliases are basically text substitutions. For example a common alias is to define alias ll="ls -l"
, but this definition will only exist for that particular shell. If you want an alias to exist in all your shells, you need to add them to your .bash_profile
or .bashrc
. Whenever you modify the .bash_profile
, it will not automatically be loaded into the shell that is already open, you either have to close the Terminal and open a new one or run source ~/.bash_profile
. Note the lack of spaces around the =
as usual when assigning values in bash. Some users like to alias the potentially dangerous commands such as rm
or mv
with the -i
option, which forces the user to confirm when a file is going to be deleted or overwritten alias rm="rm -i"
.
You can list all the defined aliases by running alias
command without any arguments, and you can unset or delete an alias with the unalias
command like unalias ll
.
The Shebang
Every script you want to run from the command line should have a shebang as the first line. A shebang looks like this #!/bin/bash
. Files are usually identified by certain codes (i.e. magic numbers) in the first few bytes of data. The hex code 23 21
converts to the ascii characters #!
and tells the system that a file is script. After the shebang comes the command that should interpret the script. Generally, any command that can interpret text files can be used. Since the value of the PATH
variable is not guaranteed in many contexts that scripts run in, the path needs to be absolute.
There are environments where you cannot predict the absolute path of a given tool. For example the bash v3.2 shell on macOS is installed by default in /bin/bash
. Users can also download and install bash version 4.4 onto their computers. The location for the the bash 4 is usually at /usr/local/bin/bash
. Since /usr/local/bin
is the first item of the default PATH
on macOS, the newer bash 4 will be chosen before the built-in bash 3.2 when the user types bash into their shell. When you use the absoute path to /bin/bash
in a shebang, you are ensuring that the provided built-in version of bash will be used. However, there are cases where you want scripts to be run with the user’s preferred tool, rather than a set path. In this case you can use #!/usr/bin/env bash
, which will determine the preferred bash tool in the user’s environment and use that to interpret the script.
sh and source
When you call sh
, you initiate a fork (sub-process) that runs a new session of /bin/sh
, which is usually a symbolic link to bash. If you launch it using ./test.sh
, the first line #!/bin/sh
would be detected, then it would be exactly the same as /bin/sh ./test.sh
. (chmod +x test.sh
to make the script executable). It executes shell scripts in a new shell process, so any variables which are assigned will disappear after the script is done. . test.sh
or source test.sh
will run the commands in the current shell (source
is a synonym for dot).
Similarly, we can tell the OS to always execute a file with node using #!/usr/bin/env node
. Here we use the env
program to find the node’s path because it’s usually in a more dynamic place, depending on your OS or node version manager.
Script arguments
A Bash script may take arguments to accomplish a task. Here’s how we can handle them in Bash and Node.js.
#!/bin/bash
echo "Script name is $0"
echo "First argument: ${1:-empty}" # `empty` is the default value in case $1 is unset
echo "All arguments: $*"
#!/usr/bin/env node
console.log(`Script name is ${process.argv[0]}`)
console.log(`First argument: ${process.argv[1] || 'empty'}`)
console.log(`All arguments: ${process.argv.join(' ')}`)
Bash Variables
Pay attention to the syntax used for assignment – foo = bar
won’t work, foo
will be interpreted as a command and = bar
as arguments.
Bash has built-in variables, like $HOME
and $BASH_VERSION
, and special parameters, like $_
which holds the last argument passed to the last executed command.
Parameter expansion is simply the usage of a value that a variable holds. Note that it won’t work within single quotes.
$ foo=bar
$ echo "Here is the content of: $foo"
Here is the content of: bar
$ echo "This is also valid: ${foo}"
This is also valid: bar
$ echo 'Here is the content of: $foo'
Here is the content of: $foo
# Command substitution: $(command) is equal to command's stdout
$ foo=$(echo bar)
$ echo "\$foo is $foo, but this gives us the same result: $(echo $foo)"
$foo is bar, but this gives us the same result: bar
# Arithmetic expressions
$ foo=$(( 1+2 ))
$ echo $foo
3
Bash Conditionals
There are some commands designed specifically to test things and return an exit status based on what they find. The first such command is test
(also known as [
, /usr/bin/[
is a program that evaluate statements). A more advanced version is [[
. [[
is much like [
, but it offers far more versatility. Read man test
for more information.
# Comparing numbers
$ num=1
$ [[ $num == 1 ]] && echo "$num is equal to 1"
$ [[ $num != 10 ]] && echo "$num is different than 10"
$ [[ $num > 0 ]] && echo "$num is greater than 0"
$ [[ $num < 10 ]] && echo "$num is less than 10"
# Comparing strings
$ foobar="foo bar"
$ #mnemonic: is length *z*ero?
$ [[ -z $foobar ]] && echo '$foobar is empty' || echo '$foobar is not empty'
$foobar is not empty
$ #mnemonic: is length *n*ot zero?
$ [[ -n $foobar ]] && echo '$foobar is not empty' || echo '$foobar is empty'
$foobar is not empty
$ [[ $foobar == "foo bar" ]] && echo "\$foobar is equal to 'foo bar'"
$foobar is equal to 'foo bar'
$ [[ $foobar =~ "^foo" ]] && echo "\$foobar starts with 'foo'"
$foobar starts with 'foo'
$ [[ $foobar =~ "bar$" ]] && echo "\$foobar ends with 'bar'"
$foobar ends with 'bar'
# Conditional blocks
if true
then
echo "This will be printed."
elif false
then
echo "This is unreachable."
else
echo "This is also unreachable."
fi
Moving to zsh
From macOS Catalina the default shell is zsh. The bash bundled with macOS has been stuck on version 3.2 for a long time now. bash v4 was released in 2009 and bash v5 in January 2019. The reason Apple has not switched to these newer versions is that they are licensed with GPL v3
. bash v3 is still GPL v2
. zsh, on the other hand, has an MIT-like license, which makes it much more palatable for Apple to include in the system by default.
Users that have /bin/bash
as their default shell on Catalina will see a prompt at the start of each Terminal session stating that zsh is now the recommended default shell. If you want to continue using /bin/bash
, you can supress this message by setting export BASH_SILENCE_DEPRECATION_WARNING=1
. You can also download and install a newer version of bash yourself. (Custom bash installations reside in a different directory usually /usr/local/bin/bash
)
The first change you will see in zsh is that the prompt looks different. zsh uses the %
character as the default prompt (of course you can change that). zsh derives from the Bourne family of shells. Because of this common ancestry, it behaves very similar in day-to-day use. There is an entire eco-system of configuration tools and themes called oh-my-zsh
which is very popular.
Hard Links and Symbolic Links
Underneath the file system, files are represented by inodes. A file in the file system is basically a link to an inode. When you delete a file, it removes the link to the underlying inode. The inode is only deleted when all links to the inode have been deleted. A hard link (ln <source> <target>
) creates another file with a link to the same underlying inode. If the real copy is deleted, the link still works because it accesses the underlying data which the real copy was accessing.
Symbolic links (i.e. etc -> private/etc
) are links to another name in the file system like shortcuts in Windows. /etc
, /tmp
and /var
are standard directories in Unix systems, but in this case these directories are actually located in /private
. You can use readlink
command to determine where a symbolic links points to. Most operations such as reading or changing are directed to the original, rather than the symbolic link. To create a symbolic link use the ln -s <source> <target>
command. The first argument is the path the symbolic link points to. If a symbolic link is deleted, its target remains unaffected. If the target is moved, renamed or deleted, the symbolic link is not automatically updated or deleted, but continues to points to the old target, a non-existing location (the link will not work).
echo 'Hello, World' > myfile.txt
ln myfile.txt my-hard-link
ln -s myfile.txt my-soft-link
In macOS Finder, you can create aliases with the menu item ‘Make Alias’ from the context menu. Finder Aliases have much the same role as symbolic links, but when the original is deleted and replaced by an item of the same name, a Finder Alias will resolve to the new item. Finder will display Aliases and symbolic links with a small arrow in the corner of the icon. Both symbolic links and Finder Aliases have a ‘Show Original’ menu item in the context menu.
The sudo
Command
Users and their access privileges control what user can read, write, or change in the system. When managing the users and their access privileges, there had to be a ‘super user’ which has access to anything. In Unix and Unix-like systems this user account is traditionally called root
.
The recommended way of gaining super user privileges from the command line is the sudo
command. The name means ‘super user do’ and will perform the command with root
privileges after verifying the user has the permission to do so. The system will prompt for your password when executing a command with sudo
. However, there is a 5 minute grace period where the sudo system caches your credentials and you do not have to re-enter the password. The Terminal prompt is set up to #
when you are running with super user privileges. To leave the root
shell, just type exit.
There is a different command which allows you to change the user: su
(short for ‘switch user’). su
will ask for credentials of the user you are switching to. So if you run su bob
, you need to have Bob’s credentials. When you run su
without a username, it assumes root
. But since logging in as root
is disabled by default on macOS, it will fail, but you can use sudo -s
or sudo -i
instead. When you run sudo -s
it will invoke a new shell running as root
. The shell that is run is the default shell of your account, and it doesn’t change the working directory. Alternatively you can use sudo -i
to invoke a root
shell. The working directory becomes /var/root
, and the shell will be /bin/sh
on macOS. It will be set up as if the root
user were logging in and will read root
’s profile.
What happens when you press a key in your terminal
Remote terminals are very old technology. In the 70s, computers were expensive. So many employees at an institution would share a single computer, and each person could have their own “terminal” to that computer.
It’s obvious that if you want to connect to a remote computer, then some information needs to be sent between the client and the server. The client needs to send the keystrokes that the user typed in, and the server needs to tell the client what to display on the screen.
For example, when you press enter
, it sends a \r
(carriage return) symbol. When you press Ctrl+C
, the client sends \x03
. If look up an ASCII table, \x03
is “End of Text”. Ctrl+D
is very similar, which sends \x04
corresponding to ASCII “End of Transmission”. What about Ctrl
+ another letter? It turns out that it’s literally just the number of that letter in the alphabet, like this Ctrl+a => \x01
, Ctrl+b => \x02
. Press Tab
, it sends \x09
, which is the ASCII code for a horizontal tab. (same as Ctrl+I
, since I
is the 9th letter)
Terminals are not just able to display black and white text; they can display colors and formatted texts thanks to escape sequences. The \x1b[
things sending to the client are called escape sequences. They change the cursor’s position, make text bold or underlined, change colours, etc.
Base64:
It’s a way of encoding arbitrary binary data in ASCII text. Essentially each 6 bits of the input is encoded in a 64-character alphabet. The “standard” alphabet uses A-Z, a-z, 0-9 and + and /, with = as a padding character. There are URL-safe variants.Base62 are
[0–9][a-z][A-Z]
, which can be used to generate short URLs. Think of the 7-bit short url (e.g. aKc3K4b), and it can be mapped to a decimal integer by using base conversion and vice versa. Practically, we can start with a counter (A large number 100000000000 in base 10 which is1L9zO9O
in Base62) and increment counter every-time we get request for new short url. This way we will always get a unique short url.Base64 is in no way meant to be a secure encryption method. It is also not a compression method. Encoding a string to Base64 typically results in 33% longer output.
Mac terminal prompt’s host becomes “bogon”
When you are doing DNS reverse lookup, the DNS server can give your machine a hostname. Normally your machine uses a router, so it has a IP like 192.168.x.x
in LAN, and this IP will be sent to the DNS server for a reverse lookup, and since it is a reserved IP address, the DNS server returns a hostname as “bogon” (an illegitimate IP address that falls into a set of IP addresses that have not been officially assigned to an entity by an internet registration institute.)
man scutil ## Manage system configuration parameters
scutil --get HostName # HostName: not set
scutil --get LocalHostName # Kexis-MacBook-Air
scutil --get ComputerName # Kexi’s MacBook Air
scutil --set HostName <newval>
scutil --set ComputerName <newval>
scutil --dns | grep nameserver # list of DNS servers configured on your system (When you use `dig` or `nslookup`)
Check if an app is running on Rosetta or M1 native architecture
Rosetta 2 enables a Mac with Apple silicon to use apps built for a Mac with an Intel processor. Rosetta is not an app that you open or interact with. Rosetta works automatically in the background whenever you use an app that was built only for Mac computers with an Intel processor.
Rosetta 2 is available only for Mac computers with Apple silicon. You’re asked to install Rosetta the first time you open an app that needs Rosetta:
- Application (Intel) means the app supports only Intel processors and needs Rosetta to work on a Mac with Apple silicon.
- Application (Universal) means the app supports both Apple silicon and Intel processors, and uses Apple silicon by default. Universal apps don’t need Rosetta.
How can you check if a Mac app runs on Rosetta or M1? Open Activity Monitor, you’ll see a column named “Kind”. If the app says “Intel” under that, you should download the native version if available.
Macs with M1 chip:
Experimental support for the M1 architecture was added in node.js v15.3 and full support was added in v16.0.
- If you have a M1 Mac, be sure you’re running node 16 or higher, otherwise you’re running via Rosetta (run
process.arch
).- If you try to install older versions of node as usual, you will probably experience either compilation errors when installing node or out-of-memory errors while running your code.
常⽤命令和作用
常⽤命令 | 作用 |
---|---|
shutdown -h now | 即刻关机 (graceful shutdown) |
reboot | 重启 |
arch | 体系结构名称 i386, i486, i586, mips, arm64, x86_64(x64 is the abbreviation for x86_64) |
sysctl -n machdep.cpu.brand_string | 查看 cpu 信息 |
uname -s | 操作系统名称 kernel name |
hostname | 计算机名 |
whoami | 当前用户名 |
who | 当前登录系统的⽤户 (The console is your physical computer and the various tty are virtual terminals) |
last | ⽤户登录⽇志 |
uptime | 系统运⾏时间、⽤户数、负载 |
env | 系统的环境变量 |
ifconfig | grep inet | IP 地址 |
ps -ax | 系统中运行的进程 (include processes not initiated by users through a terminal) |
ps -ax | grep “Visual Studio Code” | combining grep with a pipe |
top | 动态显示 cpu /内存/进程等情况 |
du -sh /dir | 指定某个⽬录的⼤⼩ |
groups | 查看所在用户组 |
find /dir -name *.bin | 在指定⽬录搜索文件 |
cat -n file1 | 查看内容并标示⾏数 |
tail -f /log | 实时查看添加到⽂件中的内容 |
grep foo hello.txt | 在⽂件中查找关键词 |
grep ^foo hello.txt | 查找以 foo 开头的内容 |
python3 -m http.server 8080 | 快速启动 http 服务 |
cat /etc/hosts | 管理 IP 地址和主机名之间的映射 |
cat /etc/resolv.conf | DNS 客户机配置文件 |
nohup ping baidu.com & | 使命令永久的在后台执行(不挂断且前台可交互) |
nohup
command allows a process to continue running in the background even after the terminal session is closed. Normally, when you close a terminal, all processes started in that terminal are terminated.nohup
prevents this.
&
at the end means run the entire command in the background as a background process.
System Information Library for Node.js: https://systeminformation.io
import { cpu, osInfo } from "systeminformation";
const { manufacturer, brand, speed, cores } = await cpu();
const { distro, release, codename} = await osInfo();
console.log(`
Node ${process.version}
${manufacturer} ${brand} ${speed}ghz ${cores} cores
${distro} ${release} ${codename}
`);
Cron Syntax
Cron is a UNIX tool that has been around for a long time, so its scheduling capabilities are powerful and proven. cRonstrue is a JavaScript library that translates cron expressions to human readable descriptions.
* * * * * *
┬ ┬ ┬ ┬ ┬ ┬
│ │ │ │ │ |
│ │ │ │ │ └ day of week (0 - 7, 1L - 7L) (0 or 7 is Sun)
│ │ │ │ └───── month (1 - 12)
│ │ │ └────────── day of month (1 - 31, L)
│ │ └─────────────── hour (0 - 23)
│ └──────────────────── minute (0 - 59)
└───────────────────────── second (0 - 59, optional)
Special characters:
*
(“all values”) - used to select all values within a field. For example, “*” in the minute field means “every minute”.?
(“no specific value”) - useful when you need to specify something in one of the two fields in which the character is allowed, but not the other. For example, if I want my trigger to fire on a particular day of the month (say, the 10th), but don’t care what day of the week that happens to be, I would put “10” in the day-of-month field, and “?” in the day-of-week field.-
- used to specify ranges. For example, “10-12” in the hour field means “the hours 10, 11 and 12”./
- used to specify increments. For example, “0/15” in the seconds field means “the seconds 0, 15, 30, and 45”. And “5/15” in the seconds field means “the seconds 5, 20, 35, and 50”.L
(“last”) - used to specify either the last day of the month, or the last xxx day of the month. L character is allowed for the day-of-month and day-of-week fields.