Internet Explorer (IE) friendly table DOM manipulation

Occasionaly we can manipulate HTML DOM easily by accessing the innerHTML of an element. In IE however, the following elements’ innerHTML are read only: COL, COLGROUP, FRAMESET, HEAD, HTML, STYLE, TABLE, TBODY, TFOOT, THEAD, TITLE, TR.

This code will fail in IE

1
tBodyEl.innerHTML = '<tr height="24px"><td colspan="7">' + 'Some Cell Value' + '</td></tr>';

A more IE friendly solution using DOM table functions.

1
2
3
4
5
var row = tBodyEl.insertRow(0);
row.setAttribute('height', '24px');
var cell = row.insertCell(0);
cell.setAttribute('colspan', '7');
cell.innerHTML = 'Some Cell Value';
Posted in javascript, Web Development | Comments Off on Internet Explorer (IE) friendly table DOM manipulation

Preventing Another Instance of A Program Running in Linux

The code below is my solution for preventing a program from running more than once at the same time in Linux.
The scenario: Consider if you have a cron job (jobA.sh) set to run at 6PM, 7PM, and 8PM.
While the 1st instance of jobA.sh started, it might (or might not) need more than an hour to finish. If it does requires more than one hour, while the 1st instance is still running, a 2nd instance of jobA.sh started.
Now you ended up with two instances of a same program consuming your CPU, files, etc – which will slow things down even more.

So here how I solve it:
When the 1st instance is started, check if there is a temporary lock file, if yes then just exits or quit. If not create the temporary lock file. And do your code/algorithm. After it finishes delete this lock file. When the 2nd instance started it will also do the same thing, checking if there is lock file.
It will be wise to put another cron job that cleans all lock file every 00:00AM. Just to be sure the next execution (at 6PM tomorrow) will be able to run

Since I’m no bash expert, let me know if there is a better way to do this. Cheers!

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#!/bin/bash
# =================================
# author: Ronald Pringadi
# sleep command is in seconds
# =================================
vLockFile=/tmp/lockfile1.lock
echo 'start'
# If lockfile does not exist do the followings
if [ ! -f $vLockFile ]; then
    touch $vLockFile
    echo "Lock file doesn't exists, executing code"
    sleep 7 # this can be your big process that takes longer than an hour, but just for proof of concept, I use 7 seconds
    rm $vLockFile
else
    echo "Lock file exists do nothing"
fi
echo 'done'
Posted in Linux, Operating System | Comments Off on Preventing Another Instance of A Program Running in Linux

Extjs – Ext.apply

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
console.clear();
var a = {
    abc : 1,
    def : 2
};
var b = {
    abc : 1.1,
    def : 2.2
};

var c = Ext.apply( a , b ); //Ext.apply overwrites 1st variable (by ref) with the 2nd var

console.log('c');
console.log(c);

console.log('a');
console.log(a);

console.log('b');
console.log(b);

Result:
c
Object { abc=1.1, def=2.2}
a
Object { abc=1.1, def=2.2}
b
Object { abc=1.1, def=2.2}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
var a = {
    abc : 1,
    def : 2
};
var b = {
    abc : 1.1,
    def : 2.2,
    gij : 3.3
};

var d = Ext.applyIf( a , b ); //Ext.applyIf overwrites 1st variable (by ref) with the 2nd var, only if it doesn't exists
console.log('d');
console.log(d);

console.log('a');
console.log(a);

console.log('b');
console.log(b);

Result:
d
Object { abc=1, def=2, gij=3.3}
a
Object { abc=1, def=2, gij=3.3}
b
Object { abc=1.1, def=2.2, gij=3.3}

Posted in Ext Js, javascript, Web Development | Comments Off on Extjs – Ext.apply

MySQL Backup and Restore – Using Command Line

Here’s how you back up a database from the command line. The first line is the backup, the second is the restore.

These commands work on both Windows and Linux. On Windows you might want to add the MySQL bin directory to your environment PATH — the easy way is Win + Break → Advanced System Settings → Environment → PATH. Then open cmd (the command prompt). If you already had a command prompt open before adding the PATH, restart it for the change to take effect.

MySQL doesn’t export stored procedures, functions, or triggers by default. If you need those, add –routines –triggers to your command. Even if you don’t currently use stored procedures, functions, or triggers, it’s safer to back up with those options enabled — they’re cheap, and you’ll thank yourself later if someone adds one.

1
2
mysqldump -uusername -ppassword --routines --triggers --lock-tables=false database_name > YOURSQLFILE.sql
mysql -uusername -ppassword database_name < YOURSQLFILE.sql

If you need to back up all databases, the user we typically use is root. This will back up all triggers and functions. You’ll need to pre-create any other users with the appropriate privileges on stored procedures or functions for the restored copy to work end-to-end.

1
2
mysqldump -uusername -ppassword --opt --events --all-databases > alldb.sql
mysql -uusername -ppassword < alldb.sql

A few useful additions.

The password-on-the-command-line problem. The -ppassword form (with no space — that’s important) is convenient but has two real issues: anyone running ps aux on the same machine can see your password while the command is running, and it gets stored in your shell history. Two cleaner options:

1
2
3
4
5
6
7
8
9
10
11
12
# Option 1: prompt for the password interactively
mysqldump -uusername -p database_name > out.sql
# (you'll be asked: Enter password:)

# Option 2: use an option file with restricted permissions
cat > ~/.my.cnf << 'EOF'
[client]
user=username
password=secret
EOF

chmod 600 ~/.my.cnf
mysqldump --defaults-file=~/.my.cnf database_name > out.sql

The ~/.my.cnf approach also helps for scheduled backups, where there’s no human around to type the password. Just make sure the file mode is 600 — MySQL clients refuse to load it otherwise on Linux.

For InnoDB tables: –single-transaction. If your tables are InnoDB (the default since MySQL 5.5), this flag wraps the dump in a transaction and gives you a consistent snapshot without locking the tables. Concurrent writes can keep happening while you back up:

1
2
mysqldump --single-transaction --routines --triggers \
    -uusername -p database_name > out.sql

Don’t combine it with –lock-tables — they fight each other. –single-transaction only works for InnoDB; if you have MyISAM tables, use –lock-tables (the default for that case) and accept that writes block.

The MySQL 8 “Access denied PROCESS privilege” trap. Starting around MySQL 5.7+/8.0, mysqldump tries to query information_schema.tablespaces and fails for non-superusers with a confusing PROCESS-privilege error:

1
mysqldump: Error: 'Access denied; you need (at least one of) the PROCESS privilege(s) for this operation' when trying to dump tablespaces

The fix is one flag — –no-tablespaces. Most application backups don’t actually need tablespace info anyway:

1
2
mysqldump --no-tablespaces --single-transaction --routines --triggers \
    -uusername -p database_name > out.sql

Compress as you go. SQL dumps compress beautifully — typical 5-10x. For anything larger than a few hundred megabytes, pipe through gzip:

1
2
3
4
5
6
7
# backup
mysqldump --single-transaction --routines --triggers -uusername -p database_name | gzip > out.sql.gz

# restore
gunzip < out.sql.gz | mysql -uusername -p database_name
# or in one shot
zcat out.sql.gz | mysql -uusername -p database_name

Restoring a single database from an –all-databases dump. Common scenario: you have a full-instance dump but only want to restore one schema. mysql has a –one-database flag that ignores statements not for the named database:

1
mysql -uusername -p --one-database wantedschema < alldb.sql

Caveat: –one-database filters by the most recent USE statement, not by table name, so it works cleanly when the dump uses USE schema; blocks (which mysqldump –all-databases emits by default). For surgical restores from arbitrary dumps, sed/awk through the file is sometimes cleaner.

For really big databases: mysql shell. If you’re operating on databases in the hundreds-of-GB range, the modern mysqlsh utility (ships with MySQL 8.0+) does parallel chunk-based dumps and is dramatically faster than mysqldump:

1
2
mysqlsh -uusername -h hostname -- util dumpInstance /path/to/backup/dir --threads=4
mysqlsh -uusername -h hostname -- util loadDump /path/to/backup/dir --threads=4

It also handles compression natively, supports resumable loads, and writes a manifest you can inspect. For everyday small-to-medium dumps, mysqldump is still simpler and the recipes above are all you need; for anything where the backup window matters, it’s worth knowing mysqlsh exists.

Posted in Database, Linux, MySQL | Comments Off on MySQL Backup and Restore – Using Command Line

Browsing without ads on mobile browsers

This post used to recommend a now-defunct Android app called “Ad Block” that turned your phone into its own proxy server. The app is gone, the Play Store link is dead, and the proxy-based approach has been superseded several times over. Mobile ad blocking is genuinely easier now than it was a decade ago — here’s what actually works today.

The four approaches, and which one fits you

Every working option boils down to one of these four:

  1. A browser that blocks ads natively. Easiest. Install the browser, you’re done.
  2. A regular browser plus a content-blocker extension. Familiar if you do this on desktop.
  3. A DNS-based blocker that works system-wide — every app, not just browsers.
  4. Network-level blocking (Pi-hole on your home router). Best in your living room, irrelevant on cellular.

Most people end up with a combination — a content-blocking browser for everyday use, plus a DNS blocker for everything else.

On Android

Browsers with built-in blocking. The simplest path:

  • Brave — Chromium-based, blocks ads and trackers by default, no setup. The most “it just works” option for someone who’s never thought about this before.
  • Vivaldi — also Chromium-based, blocking is on by default with a more granular control panel.
  • Cromite (formerly Bromite) — a privacy-hardened Chromium fork with built-in adblocking. Niche, but well-regarded by people who care about this stuff.
  • Mull — a Firefox-based browser with privacy patches and uBlock Origin pre-configured. F-Droid only, not Play Store.

Firefox for Android with extensions. Firefox is the only major mobile browser on Android that supports a real desktop-style extension system, and that’s a big deal. You can install:

  • uBlock Origin — the gold-standard content blocker. Free, open-source, no “acceptable ads” program. This is the one I’d recommend.
  • AdBlock Plus — works fine but ships with an “acceptable ads” allowlist enabled by default; you can disable it in settings if you want. The original AdBlock that this post used to recommend by name.
  • Ghostery — focused more on tracker blocking than ad blocking, with a nicer UI for seeing what each page is trying to load.

Open Firefox for Android, tap the menu, choose “Add-ons,” and install the one you want.

System-wide DNS blocking. The trick that works for every app on your phone — not just browsers — is to point your DNS at a service that returns NXDOMAIN for known ad/tracker hosts. No root required, no proxy, no app permissions to worry about:

  • NextDNS — a configurable DNS service with a generous free tier. You enable it once via Android’s Private DNS setting (Settings → Network & internet → Private DNS) and pick which blocklists to apply.
  • AdGuard DNS — similar idea, also free for personal use.
  • AdGuard for Android (the standalone app, not the DNS service) does this plus content filtering. Note: not in Play Store because Google’s policies forbid system-wide ad blockers; install via the AdGuard website.

DNS-based blocking is the closest spiritual successor to the proxy trick from this post’s original incarnation — same idea (intercept network requests before they fetch ads), better implementation (kernel-level, supports HTTPS, no app to babysit).

On iOS

iOS is more constrained, because Apple’s WebKit policy means every browser on iOS is just a Safari skin with branding. Firefox on iOS doesn’t run Gecko; Chrome doesn’t run Blink. Real desktop-style extensions don’t exist there. So the techniques are different:

  • Safari content blockers. Apple has a first-party API for this. Install 1Blocker, AdGuard for Safari, or Wipr from the App Store, then enable it in Settings → Safari → Extensions. These work in Safari only.
  • Brave on iOS — still WebKit under the hood, but Brave’s built-in shields work because they’re applied via Apple’s content-blocker API, same plumbing as the standalone blockers.
  • System-wide DNS blocking works exactly like on Android: Settings → General → VPN & Device Management → DNS, or install the NextDNS/AdGuard configuration profile. Same NXDOMAIN trick, blocks across all apps.

Firefox for iOS does not support uBlock Origin or any other extension in the desktop sense. If you came here looking for that specifically, the answer is “not possible on iOS — use a Safari content blocker plus a DNS blocker.”

What I’d actually do

If I were setting up a phone from scratch and wanted no ads with minimum fuss:

  • Android: Install Brave for casual browsing, Firefox + uBlock Origin for anything where you want fine control, and configure NextDNS or AdGuard DNS as Private DNS for system-wide coverage of in-app ads.
  • iOS: Install AdGuard for Safari (or Wipr if you want something simpler), and add the NextDNS configuration profile.

The setup is reversible and free, and you’ll save a meaningful chunk of mobile data along the way. 🎉

Two notes worth knowing. Some sites detect ad blockers and refuse to load — that’s between you and the publisher, but most blockers have a one-tap “disable on this site” button for the rare cases where you decide it’s worth it. And remember that if you’re paying for a service (a subscription, a one-time purchase), you’re often the customer and not the product; the case for blocking ads on those sites is weaker, and many will let you disable ads in settings if you ask.

Posted in Android | Comments Off on Browsing without ads on mobile browsers

Benchmark How Long a Program Runs In Linux Using Bash

The following bash code might come in handy if you want to benchmark how long a program runs. The example assumes you want to pass two parameters along to the program. A timestamp is captured at the start and end of execution, and the difference (the processing time) is printed in seconds.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#!/bin/bash
##################################################
# Benchmark the processing time when a program
# executes with two required parameters: parameter1 parameter2
##################################################

if [[ $1 = "" || $2 = "" ]]; then
        echo "Usage: `basename $0` parameter1 parameter2"
else
        _start_time=`date +%s`
        _parameter1=$1
        _parameter2=$2
        ### YOUR COMMAND HERE WITH parameter1 AND parameter2 ###
        _end_time=`date +%s`
        _processing_time=$((_end_time-_start_time))
        echo "Source File: $_parameter1"
        echo "Destination File: $_parameter2"
        echo "Start time: $_start_time"
        echo "End time: $_end_time"
        echo "Processing time is: $_processing_time"
fi

A few useful additions.

One thing to know about the script above: the ### YOUR COMMAND HERE ### line is just a comment — bash doesn’t run anything. If you copy-paste and try it, you’ll get Processing time is: 0 because the timer fires at _start_time, immediately advances to _end_time, and there’s nothing in between. Replace that comment with the actual command you want to measure, e.g.:

1
rsync -a "$_parameter1" "$_parameter2"

The simpler answer: time. If “how long did this take?” is the only question you’re asking, bash already has a builtin for it:

1
2
3
4
time ./myprogram param1 param2
# real    0m1.234s
# user    0m0.045s
# sys     0m0.012s

The three numbers are: wall-clock time (real), CPU time spent in user-mode code (user), and CPU time spent in kernel calls (sys). When user + sys is much less than real, your program was waiting on something (disk, network, sleep). When they’re roughly equal, it was CPU-bound.

The time keyword also wraps a whole pipeline or compound command:

1
time { ./step1 && ./step2 | tee out.log; }

The richer answer: /usr/bin/time. The standalone GNU time binary (different program, same name — note the explicit path or you’ll get the bash builtin) reports a lot more:

1
2
3
4
5
6
7
8
/usr/bin/time -v ./myprogram param1 param2
# Command being timed: "./myprogram"
# Maximum resident set size (kbytes): 12480
# Elapsed (wall clock) time (h:mm:ss or m:ss): 0:01.23
# Percent of CPU this job got: 87%
# Major (requiring I/O) page faults: 0
# Minor (reclaiming a frame) page faults: 1532
# ...

That’s the one to reach for when you care about memory usage, page faults, or context switches alongside timing. Some distros don’t ship it by default — apt install time or dnf install time if it’s missing.

Sub-second timing without date +%s. The original script’s resolution is one second, which is too coarse for anything fast. A few ways to get finer numbers:

1
2
3
4
5
6
7
8
9
10
11
# Nanosecond resolution via date (still spawns a subprocess each call)
_start=$(date +%s%N)
# ... work ...
_end=$(date +%s%N)
echo "$(( (_end - _start) / 1000000 )) ms"

# Bash 5+ has $EPOCHREALTIME — sub-second, no subprocess at all
_start=$EPOCHREALTIME
# ... work ...
_end=$EPOCHREALTIME
awk -v s="$_start" -v e="$_end" 'BEGIN { printf "%.3f s\n", e - s }'

$EPOCHREALTIME is a string like 1715000123.456789 with microsecond resolution. It’s free — no fork, no exec — so it’s safe to call inside a hot loop. The catch: bash arithmetic is integer-only, so you need awk or bc to do the floating-point subtraction.

When you want statistical confidence: hyperfine. If the question is “is my new version actually faster?” — you want averages over many runs, with warmup, and ideally a confidence interval. hyperfine is the modern tool for this:

1
2
3
4
5
6
hyperfine --warmup 3 --runs 20 \
    './build_old.sh input.txt' \
    './build_new.sh input.txt'
# Summary
#   './build_new.sh input.txt' ran
#     1.42 ± 0.03 times faster than './build_old.sh input.txt'

It runs each command repeatedly, discards the warmup runs (which are usually slower because of cold caches), reports mean ± standard deviation, and gives you a relative-speedup figure. If you’re benchmarking anything where one-off variance matters — and that’s most things — this beats running time three times and squinting at the numbers.

Two gotchas worth knowing.

  • System clock changes break date +%s. If NTP slews the clock backwards mid-run, your _processing_time can be negative or inflated. $EPOCHREALTIME has the same problem (both read CLOCK_REALTIME). For benchmarking, the correct clock is CLOCK_MONOTONIC, which never goes backwards — but bash doesn’t expose it directly. Use time or hyperfine for anything that needs to survive a clock change.
  • Disk caches make repeated runs misleadingly fast. The first run of a program reads cold data from disk; the second run hits the page cache and looks 10x faster. Either drop caches between runs (echo 3 | sudo tee /proc/sys/vm/drop_caches on Linux) or — easier — let hyperfine handle the warmup phase for you.
Posted in Linux, Operating System, Ubuntu | Comments Off on Benchmark How Long a Program Runs In Linux Using Bash

Auto Login Using SSH Public and Private Keys

2026 update: This post originally used ssh-keygen -t rsa. The current recommendation (e.g. GitHub’s SSH key guide) is Ed25519 — it produces shorter keys, is faster, and is considered more secure. Use RSA (4096-bit) only as a fallback for systems that don’t support Ed25519.

The -C “your_email@example.com” flag is just a comment/label embedded in the public key — it has no effect on the cryptography. It’s helpful for identifying which key is which when you have several. You can put any string there (or omit -C entirely). 🔐


Assume that your main server (the one you use the most) is ServerA. And from ServerA, you want to automatically login to ServerB. For illustration purpose, ServerB can be a repo server and you don’t want to keep being asked for password everytime you want to commit files.

1. At ServerA issue:

1
ssh-keygen -t ed25519 -C "your_email@example.com"

This will create a public + private key for ServerA.

2. Enter (empty) for passphrase

1
2
Your identification has been saved in /home/ronald/.ssh/id_ed25519.
Your public key has been saved in /home/ronald/.ssh/id_ed25519.pub.

3. Copy over the newly created public key from ServerA to ServerB:

1
scp .ssh/id_ed25519.pub ronald@ServerB:~/id_ed25519.pub.ServerA

Note that once the file arrives at ServerB, it named as id_ed25519.pub.ServerA (not id_ed25519.pub), and it will be stored at ronald’s home directory at ServerB.

4. Ssh to ServerB, and issue this command:

1
cat id_ed25519.pub.ServerA >> .ssh/authorized_keys

ServerA public key will be merged/appended to ServerB authorized_keys file.

This way everytime you want to access ServerB from ServerA, you won’t be asked for a password. 🎉

Posted in Linux, Operating System, Ubuntu | Comments Off on Auto Login Using SSH Public and Private Keys

Linux symlinks: creating, deleting, and the gotchas worth knowing

The basic command for creating a symbolic link (a “symlink” or “soft link”) to a folder or file is ln -s:

1
2
3
4
5
6
7
8
ln -s /home/ronald/somefolder /home/ronald/newfolder

# or if your current directory is already /home/ronald:
cd /home/ronald
ln -s somefolder newfolder

# general form:
ln -s existingSourceFolder newLinkedFolder

The first argument is the target (what the link points at), the second is the link name (what gets created). I get this order backwards every time — the mnemonic that finally stuck is “ln -s is like cp: source first, destination second.”

To remove all symlinks in the current directory (and only symlinks — leaving regular files and directories untouched):

1
find . -maxdepth 1 -type l -exec rm -f {} \;

-type l is the filter that matches symlinks specifically. -maxdepth 1 keeps the search from descending into subdirectories, so you only kill the symlinks at the top level.


A few useful additions: symlinks have sharp edges.

Symlinks look simple — “a file that points to another file” — but they have a handful of behaviors that catch people out. The ones below are the ones I’ve actually been bitten by.

Symlinks vs. hard links. The -s in ln -s is the difference. Without it, you get a hard link: a second directory entry pointing to the same inode. Hard links don’t break if you move or rename the original (because both names refer to the same data). Symlinks store a path and break if the target moves. Hard links can’t span filesystems and can’t point to directories (the kernel forbids it to prevent loops); symlinks can do both.

Inspecting a symlink. A few different tools answer different questions:

1
2
3
4
5
ls -l newfolder              # shows: newfolder -> /home/ronald/somefolder
readlink newfolder           # prints just the target: /home/ronald/somefolder
readlink -f newfolder        # follows all symlinks, prints absolute target
realpath newfolder           # same idea as readlink -f, slightly different edge cases
stat newfolder               # full metadata; "File: newfolder -> ..."

readlink -f is the one I reach for most — it tells you where a symlink ultimately resolves, even through chains of symlinks-pointing-to-symlinks.

Relative vs. absolute targets. The path you pass as the target is stored verbatim. If you write a relative target, it’s resolved relative to the directory the link lives in, not the directory you ran the command from. This trips people up:

1
2
3
4
5
6
cd /tmp
ln -s ../etc/hosts mylink         # mylink points to ../etc/hosts
                                  # which from /tmp resolves to /etc/hosts ✓

mv mylink /var/log/               # now /var/log/mylink points to ../etc/hosts
                                  # which resolves to /var/etc/hosts ✗ broken

Rule of thumb: use absolute paths unless you specifically want the link to follow when both source and link move together (common pattern: shipping a relative symlink inside a tarball or git repo).

The rm -rf on a symlink-to-directory trap. This one is genuinely dangerous:

1
2
3
4
ln -s /important/data ./mydata
rm -rf mydata                # removes the symlink only — /important/data is safe
rm -rf mydata/               # WITH trailing slash: rm follows the link and
                             # recursively deletes /important/data

The trailing slash changes the command from “delete this symlink” to “delete what’s inside the directory the symlink points to.” If you’re in the habit of tab-completing paths, your shell often appends the slash for you. Be deliberate when removing symlinks to directories.

Updating an existing symlink. By default ln -s refuses to overwrite. The flags vary depending on what you’re replacing:

1
2
3
4
ln -sf newtarget mylink           # works for symlinks to files
ln -sfn newtarget mydirlink       # for symlinks to directories — without -n,
                                  # ln creates a NEW link INSIDE the existing
                                  # directory link instead of replacing it

Forgetting -n on a directory symlink is one of those bugs that’s invisible until you wonder why your link “didn’t take” — and meanwhile a stray newtarget entry is hiding in the linked directory.

Finding broken symlinks. Symlinks happily survive their targets being deleted. To find ones that no longer point anywhere valid:

1
2
find . -xtype l                          # broken symlinks under current dir
find / -xtype l 2>/dev/null              # the same, system-wide (slow)

-xtype l means “the symlink’s target is not a real file/directory we can stat” — i.e. the link is dangling. Useful as a periodic sanity check after you’ve moved or removed a directory that other things might have linked to.

Finding what links to a given target. The reverse question — “is anything pointing at this file?” — has no fast answer; you have to scan:

1
2
find / -lname "*somefolder*" 2>/dev/null    # by target pattern
find / -samefile /home/ronald/somefolder 2>/dev/null  # exact same inode

The first form matches the symlink’s stored target string; the second works for hard links (same inode). There’s no equivalent of “give me everything that resolves to this path,” because resolving every symlink on the system would be expensive — you’d be running readlink -f on every link in the search tree.

Posted in Linux, Operating System, Ubuntu | Comments Off on Linux symlinks: creating, deleting, and the gotchas worth knowing

A Linux primer: identifying your distro and other commands that work everywhere

Here are a few commands you can use to find out which Linux distribution you’re on — pick whichever your system has:

1
2
3
4
5
6
7
cat /etc/issue

cat /proc/version

dmesg | head -1

cat /etc/*-release

Of those, cat /etc/*-release is the most reliable on modern systems. The wildcard catches both /etc/os-release (the systemd-mandated standard, present on virtually all distros since around 2012) and any distro-specific files like /etc/redhat-release, /etc/lsb-release, or /etc/debian_version — so you don’t need to know in advance what the file is called.

If you only want a clean machine-readable answer and don’t care about the others, source /etc/os-release and read its variables directly:

1
2
. /etc/os-release
echo "$NAME $VERSION_ID"   # e.g. "Ubuntu 22.04"

A few useful additions: commands that work the same across distros.

One of the joys of Linux is that once you’ve learned the basics, the same commands work whether you’re on Ubuntu, Fedora, Arch, Alpine, or Amazon Linux. Here are some that show up in almost every troubleshooting session:

What kernel and architecture am I on?

1
2
3
uname -a              # full kernel info: kernel name, hostname, release, arch
uname -r              # just the kernel release (useful for matching kernel modules)
arch                  # CPU architecture: x86_64, aarch64, armv7l, etc.

How much memory and disk do I have?

1
2
3
free -h               # memory: human-readable units
df -h                 # disk usage by filesystem
du -sh /var/log       # size of a specific directory

What’s running and using resources?

1
2
3
ps aux                # all processes, BSD-style output
top                   # interactive process viewer (q to quit)
htop                  # nicer interactive viewer (often needs install)

Who am I, where am I, what’s the time?

1
2
3
4
5
6
whoami                # current user
id                    # user + group IDs
hostname              # machine name
pwd                   # current directory
date                  # current date and time, with timezone
uptime                # how long the system has been running, plus load average

Networking basics. The classic ifconfig and netstat are deprecated in favor of ip and ss, which ship with iproute2 and are present on every modern distro:

1
2
3
4
5
ip addr               # network interfaces and their IPs (replaces ifconfig)
ip route              # routing table (replaces route)
ss -tulpn             # listening TCP/UDP sockets with PIDs (replaces netstat)
ping -c 4 example.com # send 4 ICMP packets and stop
curl -I https://example.com  # fetch just response headers

Which desktop am I running? If you ssh into a graphical box and aren’t sure whether it’s GNOME, KDE Plasma, XFCE, Cinnamon, or something else, two environment variables will tell you:

1
2
echo $XDG_CURRENT_DESKTOP    # GNOME, KDE, XFCE, MATE, X-Cinnamon, etc.
echo $DESKTOP_SESSION        # the session name (often more specific)

Both are set by the display manager when you log in graphically; they’re empty if you’re on a headless box or in a plain ssh shell that didn’t inherit them. If they come back empty but you know there’s a desktop running, this works too:

1
ps -e | grep -Eo 'gnome-shell|plasmashell|xfce4-session|cinnamon|mate-session' | head -1

X11 or Wayland? Once you’ve got a desktop, the next question is which display server it’s drawing through. The simplest check is the session-type variable, set by systemd’s logind:

1
echo $XDG_SESSION_TYPE       # "x11" or "wayland"

If that’s empty (again, common over plain ssh), use loginctl to ask the session manager directly:

1
2
loginctl show-session $(loginctl | awk 'NR==2 {print $1}') -p Type
# Type=wayland

And if you want to see whether the Wayland or X11 socket actually exists on the system:

1
2
ls /run/user/$(id -u)/wayland-*    # Wayland socket(s) for your user
ls /tmp/.X11-unix/                  # X11 sockets (one per :display)

Why this matters: some tools behave differently or don’t work at all under Wayland (screenshot utilities, screen sharing, global hotkey daemons, automation tools like xdotool). When something graphical “just doesn’t work” on a modern Linux desktop, “are you on Wayland?” is often the first question to ask.

Looking for files and text.

1
2
3
4
find / -name "*.conf" 2>/dev/null      # find files by name
grep -rni "error" /var/log              # recursive case-insensitive search
locate sshd_config                      # very fast — uses an index, may need updatedb
which python3                           # path to a command in $PATH

Permissions and ownership.

1
2
3
4
ls -la                                  # detailed listing with permissions
stat /etc/passwd                        # all metadata for a file
chmod 644 file.txt                      # rw for owner, r for group/others
chown user:group file.txt               # change owner and group

Service management. Systemd is the init system on virtually every mainstream distro now (Debian, Ubuntu, RHEL/Fedora, SUSE, Arch). The same systemctl commands work everywhere:

1
2
3
4
5
6
systemctl status nginx                  # is it running?
systemctl start nginx
systemctl stop nginx
systemctl enable nginx                  # auto-start at boot
systemctl restart nginx
journalctl -u nginx -f                  # tail the service's logs

Alpine and a few minimalist distros use OpenRC instead (rc-service, rc-update) — worth knowing when you ssh into a small container.

Where the distros do diverge: package management. This is the one place you have to know which family you’re on:

1
2
3
4
5
Debian / Ubuntu      apt install <pkg>     apt update / apt upgrade
RHEL / Fedora / CentOS dnf install <pkg>   dnf update          (yum on older releases)
openSUSE             zypper install <pkg>  zypper refresh / zypper update
Arch                 pacman -S <pkg>       pacman -Syu
Alpine               apk add <pkg>         apk update / apk upgrade

If a script needs to work across families, the cleanest pattern is to detect $ID from /etc/os-release and dispatch:

1
2
3
4
5
6
7
8
. /etc/os-release
case "$ID" in
    ubuntu|debian)        apt install -y curl ;;
    rhel|centos|fedora)   dnf install -y curl ;;
    arch|manjaro)         pacman -S --noconfirm curl ;;
    alpine)               apk add curl ;;
    *)                    echo "unsupported distro: $ID"; exit 1 ;;
esac

That’s the foundation. Once you have these committed to muscle memory, ssh-ing into an unfamiliar Linux box stops being intimidating — the same handful of commands answer the same handful of questions everywhere.

Posted in Linux, Operating System, Ubuntu | Comments Off on A Linux primer: identifying your distro and other commands that work everywhere

How to Set up an FTP Server in Ubuntu Linux

Login to your Linux shell menu as root

1
2
[root@locahost]# apt-get install vsftpd
[root@locahost]# vim /etc/vsftpd.conf

At VIM, 1st comment out anonymous_enable by adding a # sign at the beginning of the line
# anonymous_enable=YES
2nd remove comment at local enable by removing the # sign
local_enable=YES

1
[root@locahost]# /etc/init.d/vsftpd restart
Posted in Linux, Operating System, Ubuntu | Comments Off on How to Set up an FTP Server in Ubuntu Linux