diff --git a/sql/hpr-db-part-14.sql b/sql/hpr-db-part-14.sql
index 71c0664..1663b06 100644
--- a/sql/hpr-db-part-14.sql
+++ b/sql/hpr-db-part-14.sql
@@ -485,7 +485,11 @@ INSERT INTO `eps` (`id`, `date`, `title`, `duration`, `summary`, `notes`, `hosti
(3985,'2023-11-10','Bash snippet - be careful when feeding data to loops',1644,'A loop in a pipeline runs in a subshell','\nOverview
\nRecently Ken Fallon did a show on HPR, number\n3962, in which he used a Bash\npipeline of multiple commands feeding their output into a\nwhile
loop. In the loop he processed the lines produced by\nthe pipeline and used what he found to download audio files belonging to\na series with wget
.
\nThis was a great show and contained some excellent advice, but the\nuse of the format:
\npipeline | while read variable; do ...
\nreminded me of the \"gotcha\" I mentioned in my own show\n2699.
\nI thought it might be a good time to revisit this subject.
\nSo, what\'s the problem?
\nThe problem can be summarised as a side effect of pipelines.
\nWhat are pipelines?
\nPipelines are an amazingly useful feature of Bash (and other shells).\nThe general format is:
\ncommand1 | command2 ...
\nHere command1
runs in a subshell and produces output (on\nits standard output) which is connected via the pipe symbol\n(|
) to command2
where it becomes its\nstandard input. Many commands can be linked together in this\nway to achieve some powerful combined effects.
\nA very simple example of a pipeline might be:
\n$ printf 'World\nHello\n' | sort\nHello\nWorld
\nThe printf
command (≡\'command1\'
) writes two\nlines (separated by newlines) on standard output and this is\npassed to the sort
command\'s standard input\n(≡\'command2\'
) which then sorts these lines\nalphabetically.
\nCommands in the pipeline can be more complex than this, and in the\ncase we are discussing we can include a loop command such as\nwhile
.
\nFor example:
\n$ printf 'World\nHello\n' | sort | while read line; do echo "($line)"; done\n(Hello)\n(World)
\nHere, each line output by the sort
command is read into\nthe variable line
in the while
loop and is\nwritten out enclosed in parentheses.
\nNote that the loop is written on one line. The semi-colons are used\ninstead of the equivalent newlines.
\nVariables and subshells
\nWhat if the lines output by the loop need to be numbered?
\n$ i=0; printf 'World\nHello\n' | sort | while read line; do ((i++)); echo "$i) $line"; done\n1) Hello\n2) World
\nHere the variable \'i\'
is set to zero before the\npipeline. It could have been done on the line before of course. In the\nwhile
loop the variable is incremented on each iteration\nand included in the output.
\nYou might expect \'i\'
to be 2 once the loop exits but it\nis not. It will be zero in fact.
\nThe reason is that there are two \'i\'
variables. One is\ncreated when it\'s set to zero at the start before the pipeline. The\nother one is created in the loop as a \"clone\". The expression:
\n((i++))
\nboth creates the variable (where it is a copy of the one in the\nparent shell) and increments it.
\nWhen the subshell in which the loop runs completes, it will delete\nthis version of \'i\'
and the original one will simply\ncontain the zero that it was originally set to.
\nYou can see what happens in this slightly different example:
\n$ i=1; printf 'World\nHello\n' | sort | while read line; do ((i++)); echo "$i) $line"; done\n2) Hello\n3) World\n$ echo $i\n1
\nThese examples are fine, assuming the contents of variable\n\'i\'
incremented in the loop are not needed outside it.
\nThe thing to remember is that the same variable name used in a\nsubshell is a different variable; it is initialised with the value of\nthe \"parent\" variable but any changes are not passed back.
\nHow to avoid the\nloss of changes in the loop
\nTo solve this the loop needs to be run in the original shell, not a\nsubshell. The pipeline which is being read needs to be attached to the\nloop in a different way:
\n$ i=0; while read line; do ((i++)); echo "$i) $line"; done < <(printf 'World\nHello\n' | sort)\n1) Hello\n2) World\n$ echo $i\n2
\nWhat is being used here is process\nsubstitution. A list of commands or pipelines are enclosed with\nparentheses and a \'less than\'
sign prepended to the list\n(with no intervening spaces). This is functionally equivalent to a\n(temporary) file of data.
\nThe redirection feature allows for data being read from a\nfile in a loop. The general format of the command is:
\nwhile read variable\n do\n # Use the variable\n done < file
\nUsing process substitution instead of a file will achieve what is\nrequired if computations are being done in the loop and the results are\nwanted after it has finished.
\nBeware of this type of\nconstruct
\nThe following one-line command sequence looks similar to the version\nusing process substitution, but is just another form of pipeline:
\n$ i=0; while read line; do echo $line; ((i++)); done < /etc/passwd | head -n 5; echo $i\nroot:x:0:0:root:/root:/bin/bash\ndaemon:x:1:1:daemon:/usr/sbin:/usr/sbin/nologin\nbin:x:2:2:bin:/bin:/usr/sbin/nologin\nsys:x:3:3:sys:/dev:/usr/sbin/nologin\nsync:x:4:65534:sync:/bin:/bin/sync\n0
\nThis will display the first 5 lines of the file but does it by\nreading and writing the entire file and only showing the first 5 lines\nof what is written by the loop.
\nWhat is more, because the while
is in a subshell in a\npipeline changes to variable \'i\'
will be lost.
\nAdvice
\n\nUse the pipe-connected-to-loop layout if you\'re aware of\nthe pitfalls, but will not be affected by them.
\nUse the read-from-process-substitution format if you\nwant your loop to be complex and to read and write variables in the\nscript.
\nPersonally, I always use the second form in scripts, but if I\'m\nwriting a temporary one-line thing on the command line I usually use the\nfirst form.
\n
\nTracing pipelines (advanced)
\nI have always wondered about processes in Unix. The process you log\nin to, normally called a shell runs a command language\ninterpreter that executes commands read from the standard input or\nfrom a file. There are several such interpreters available, but we\'re\ndealing with bash here.
\nProcesses are fairly lightweight entities in Unix/Linux. They can be\ncreated and destroyed quickly, with minimal overhead. I used to work\nwith Digital Equipment Corporation\'s OpenVMS operating system\nwhich also uses processes - but these are much more expensive to create\nand destroy, and therefore slow and less readily used!
\nBash pipelines, as discussed, use subshells. The description\nin the Bash man page says:
\n\nEach command in a multi-command pipeline, where pipes are created, is\nexecuted in a subshell, which is a separate process.
\n
\nSo a subshell in this context is basically another child\nprocess of the main login process (or other parent process), running\nBash.
\nProcesses (subshells) can be created in other ways. One is to place a\ncollection of commands in parentheses. These can be simple Bash\ncommands, separated by semi-colons, or pipelines. For example:
\n$ (echo "World"; echo "Hello") | sort\nHello\nWorld
\nHere the strings \"World\"
and \"Hello\"
, each\nfollowed by a newline are created in a subshell and written to standard\noutput. These strings are piped to sort
and the end result\nis as shown.
\nNote that this is different from this example:
\n$ echo "World"; echo "Hello" | sort\nWorld\nHello
\nIn this case \"World\"
is written in a separate command,\nthen \"Hello\"
is written to a pipeline. All\nsort
sees is the output from the second echo
,\nwhich explains the output.
\nEach process has a unique numeric id value (the process id\nor PID). These can be seen with tools like ps
or\nhtop
. Each process holds its own PID in a Bash variable\ncalled BASHPID
.
\nKnowing all of this I decided to modify Ken\'s script from show\n3962 to show the processes being created - mainly for my interest,\nto get a better understanding of how Bash works. I am including it here\nin case it may be of interest to others.
\n#!/bin/bash\n\nseries_url="https://hackerpublicradio.org/hpr_mp3_rss.php?series=42&full=1&gomax=1"\ndownload_dir="./"\n\npidfile="/tmp/hpr3962.sh.out"\ncount=0\n\necho "Starting PID is $BASHPID" > $pidfile\n\n(echo "[1] $BASHPID" >> "$pidfile"; wget -q "${series_url}" -O -) |\\n (echo "[2] $BASHPID" >> "$pidfile"; xmlstarlet sel -T -t -m 'rss/channel/item' -v 'concat(enclosure/@url, "→", title)' -n -) |\\n (echo "[3] $BASHPID" >> "$pidfile"; sort) |\\n while read -r episode; do\n\n [ $count -le 1 ] && echo "[4] $BASHPID" >> "$pidfile"\n ((count++))\n\n url="$( echo "${episode}" | awk -F '→' '{print $1}' )"\n ext="$( basename "${url}" )"\n title="$( echo "${episode}" | awk -F '→' '{print $2}' | sed -e 's/[^A-Za-z0-9]/_/g' )"\n #wget "${url}" -O "${download_dir}/${title}.${ext}"\n done\n\necho "Final value of \$count = $count"\necho "Run 'cat $pidfile' to see the PID numbers"
\nThe point of doing this is to get information about the pipeline\nwhich feeds data into the while loop
. I kept the rest\nintact but commented out the wget
command.
\nFor each component of the pipeline I added an echo
\ncommand and enclosed it and the original command in parentheses, thus\nmaking a multi-command process. The echo
commands write a\nfixed number so you can tell which one is being executed, and it also\nwrites the contents of BASHPID
.
\nThe whole thing writes to a temporary file\n/tmp/hpr3962.sh.out
which can be examined once the script\nhas finished.
\nWhen the script is run it writes the following:
\n$ ./hpr3962.sh\nFinal value of $count = 0\nRun 'cat /tmp/hpr3962.sh.out' to see the PID numbers
\nThe file mentioned contains:
\nStarting PID is 80255\n[1] 80256\n[2] 80257\n[3] 80258\n[4] 80259\n[4] 80259
\nNote that the PID values are incremental. There is no guarantee that\nthis will be so. It will depend on whatever else the machine is\ndoing.
\nMessage number 4 is the same for every loop iteration, so I stopped\nit being written after two instances.
\nThe initial PID is the process running the script, not the login\n(parent) PID. You can see that each command in the pipeline runs in a\nseparate process (subshell), including the loop.
\nGiven that a standard pipeline generates a process per command, I was\nslightly surprised that the PID numbers were consecutive. It seems that\nBash optimises things so that only one process is run for each element\nof the pipe. I expect that it would be possible for more processes to be\ncreated by having pipelines within these parenthesised lists, but I\nhaven\'t tried it!
\nI found this test script quite revealing. I hope you find it useful\ntoo.
\nLinks
\n\n\n\n- Bash process substitution:\n
\n
\n\n- HPR shows referenced:\n
\n
\n\n',225,42,1,'CC-BY-SA','Bash,loop,process,shell',0,0,1),
(3992,'2023-11-21','Test recording on a wireless mic',223,'Archer72 tests out a wireless mic with a USB C receiver','
LEKATO 2\nPack Wireless Microphone with Charging Case
\nhttps://www.amazon.com/gp/product/B0C4SNT6QK
\n\n- USB C
\n- Two microphones in a charging case
\n- Charge phone and use the receiver simultaneously
\n
\nClaims
\n\n- 75 ft. transmission range
\n- Wireless Mic can work continuously for 5 hours, and the charging box\ncan quickly charge the device 4 times. The total usage time reaches 25\nhours
\n
\nAxet Audio recorder on\nF-Droid
\nhttps://f-droid.org/packages/com.github.axet.audiorecorder
\n\n- Works to record stereo with this mic set
\n
\n',318,0,0,'CC-BY-SA','Recording, Microphone, Wireless, USB \'C\', F-droid, Android App',0,0,1),
(4221,'2024-10-07','HPR Community News for September 2024',0,'HPR Volunteers talk about shows released and comments posted in September 2024','',159,47,1,'CC-BY-SA','Community News',0,0,1),
-(4241,'2024-11-04','HPR Community News for October 2024',0,'HPR Volunteers talk about shows released and comments posted in October 2024','',159,47,1,'CC-BY-SA','Community News',0,0,1);
+(4241,'2024-11-04','HPR Community News for October 2024',0,'HPR Volunteers talk about shows released and comments posted in October 2024','',159,47,1,'CC-BY-SA','Community News',0,0,1),
+(3986,'2023-11-13','Optical media is not dead',435,'Archer72 shows command line options for creating and writing iso files','Brought up by Klaatu\non GnuWorldOrder.info
\n\n4.7Gb DVD - Actual capacity: 4.377Gb
\nWhat is the\nactual storage capacity of a dvd disc
\nA disc with a 25GB capacity is the equivalent of 23.28 gigabytes\nNormally rated at 50GB, in practice they can record about 46.57GB of\ndata
\nWhat is the\nmaximum capacity of a blu ray disc
\nGenerate ISO image from directory
\ngenisoimage -U -R -allow-lowercase -allow-multidot -o mydvd.iso "$1"
\nAlternative is mkisofs
\n-U\n\nAllows "untranslated" filenames, completely violating the ISO9660 standards described above. Enables the following flags: -d -l -N -allow-leading-dots -relaxed-filenames -allow-lowercase -allow-multidot -no-iso-translate. Allows more than one `.' character in the filename, as well as mixed-case filenames. This is useful on HP-UX, where the built-in cdfs filesystem does not recognize any extensions. Use with extreme caution.\n\n-R\n\nGenerate SUSP and RR records using the Rock Ridge protocol to further describe the files on the ISO9660 filesystem.\n\n[Wikipedia - ISO 9660](https://en.wikipedia.org/wiki/ISO_9660#SUSP "Wikipedia - ISO 9660")
\nCorrected command
\ngenisoimage -R -o mydvd.iso "$1"
\nBurning data to a DVD or Blu-ray
\nNote:
\nMake sure that the medium is not mounted when you begin to write to\nit. Mounting may happen automatically if the medium contains a readable\nfile system. In the best case, it will prevent the burn programs from\nusing the burner device. In the worst case, there will be misburns\nbecause read operations disturbed the drive. So if in doubt, do:
\numount /dev/sr0
\ngrowisofs has a small bug with blank BD-R media. It issues an error\nmessage after the burning is complete. Programs like k3b then believe\nthe whole burn run failed. To prevent this, either format the blank BD-R\nby dvd+rw-format /dev/sr0 before submitting it to growisofs or use\ngrowisofs option
\n-use-the-force-luke=spare:none
\nArchwiki - Optical disk\nburning
\nBurning an ISO image to\nCD, DVD, or BD
\nTo\nburn a readily prepared ISO image file isoimage.iso onto an optical\nmedium, run for CD:
\ncdrecord -v -sao dev=/dev/sr0 isoimage.iso
\nand for DVD or BD:
\ngrowisofs -dvd-compat -Z /dev/sr0=isoimage.iso
\nfor CD, DVD, BD:
\nxorriso -as cdrecord -v dev=/dev/sr0 -dao isoimage.iso
\nOther reading
\nArchwiki - Optical disk\ndrive
\nArchwiki - Optical disk\ndrive
\nDebian - Gensisoimage man\npage
\nDebian - Gensisoimage man\npage
\nDebian wiki -\ngenisoimage and xorrisofs
\nDebian wiki
\nArchiving data on Blu-ray\ndiscs
\nArchiving data
\nMount\nan ISO file and Burning it to CD-R/DVD-R/BluRay in Linux
\nMount an ISO file and\nBurning
\nWhat\nis the size capacity of my DVD, Dual Layer DVD or Blu-ray disc?
\nWhat is the size capacity of\nmy DVD
\nWikipedia ISO9660
\nWikipedia ISO9660
\n',318,0,0,'CC-BY-SA','Command line, Create ISO, Burn ISO, Optical media, DVD, CD, Blu-ray',0,0,0),
+(3987,'2023-11-14','The Grim Dawn',2391,'Sgoti rambles about a video game called Grim Dawn.','\nThis work is licensed under a Creative Commons\nAttribution-ShareAlike 4.0 International License.
\n',391,0,0,'CC-BY-SA','Grim Dawn, ARPG, Diablo 4, Video Games',0,0,0),
+(3988,'2023-11-15','Beeper.com',750,'operat0r talks about Beeper dot com a multi chat client','I talk about Beeper dot com a multi chat client
\nBeeper is a universal messaging app that lets you chat with anyone on\nany chat app, including Whatsapp, iMessage, Telegram and 12 other\nnetworks.
\nLinks
\n\n',36,0,1,'CC-BY-SA','chat,messaging,mobile',0,0,0),
+(3989,'2023-11-16','LastPass Security Update 1 November 2023',553,'LastPass was hacked, what should you do?','In 2022, LastPass disclosed that it had been hacked, and I think by\nnow just about everyone has heard about it. Now we have evidence that\npassword vaults have been hacked. So what does this mean, and what\nshould you do?
\nLinks:
\n\n',198,74,0,'CC-BY-SA','LastPass, password vault',0,0,0);
/*!40000 ALTER TABLE `eps` ENABLE KEYS */;
UNLOCK TABLES;
@@ -994,7 +998,3 @@ INSERT INTO `miniseries` (`id`, `name`, `description`, `private`, `image`, `vali
(25,'Programming 101','A series focusing on concepts and the basics of programming',0,'',1),
(26,'RoundTable','Panelists dicuss a topic each month.',0,'',1),
(82,'Vim Hints','\r\nVarious contributors lead us on a journey of discovery of the Vim (and vi) editors.\r\n
\r\n\r\nVim is a highly configurable text editor built to enable efficient text editing. It is an improved version of the vi editor distributed with most UNIX systems.\r\n
\r\n\r\nhttps://www.vim.org/about.php\r\n
',0,'',1),
-(28,'NewsCast','What\'s happening in the News world',1,'',1),
-(29,'How I got into tech','Started by monsterb, this series invites people to share with us how they found Linux. It has become traditional for first time hosts to share with us their journey to Linux. Indeed it has morphed to be way to share your journey in tech right up to your first contribution to HPR.',0,'',1),
-(30,'Tit Radio','Welcome to TiT Radio! The only Hacker Public Radio show with super cow powers broadcasting live on ddphackradio.org every utter Saturday night at 11pm CST. You may be asking yourself \"What in tarnation is Tit Radio?\" Well, it\'s a potluck style roundtable of geeks talking about Free Software, GNU + Linux, and anything geeky the TiTs bring to the table. Chat with the TiTs over at irc.freenode.net #linuxcranks. Thats no bull.',1,'',1),
-(34,'Talk Geek to me','deepgeek talks geek to his fans',1,'',1),
diff --git a/sql/hpr-db-part-15.sql b/sql/hpr-db-part-15.sql
index 3858ffa..5c374b2 100644
--- a/sql/hpr-db-part-15.sql
+++ b/sql/hpr-db-part-15.sql
@@ -1,3 +1,7 @@
+(28,'NewsCast','What\'s happening in the News world',1,'',1),
+(29,'How I got into tech','Started by monsterb, this series invites people to share with us how they found Linux. It has become traditional for first time hosts to share with us their journey to Linux. Indeed it has morphed to be way to share your journey in tech right up to your first contribution to HPR.',0,'',1),
+(30,'Tit Radio','Welcome to TiT Radio! The only Hacker Public Radio show with super cow powers broadcasting live on ddphackradio.org every utter Saturday night at 11pm CST. You may be asking yourself \"What in tarnation is Tit Radio?\" Well, it\'s a potluck style roundtable of geeks talking about Free Software, GNU + Linux, and anything geeky the TiTs bring to the table. Chat with the TiTs over at irc.freenode.net #linuxcranks. Thats no bull.',1,'',1),
+(34,'Talk Geek to me','deepgeek talks geek to his fans',1,'',1),
(35,'SELF Talks 2009','South East Linux Fest talks 2009',1,'',1),
(36,'Software Freedom Day Dundee 2009','Software Freedom Day Dundee 2009',1,'',1),
(38,'A Little Bit of Python','\r\nInitially based on the podcast \"A Little Bit of Python\", by Michael Foord, Andrew Kuchling, Steve Holden, Dr. Brett Cannon and Jesse Noller. https://www.voidspace.org.uk/python/weblog/arch_d7_2009_12_19.shtml#e1138\r\n
\r\n\r\nNow the series is open to all.\r\n
',0,'',1),
@@ -409,4 +413,4 @@ UNLOCK TABLES;
/*!40014 SET UNIQUE_CHECKS=@OLD_UNIQUE_CHECKS */;
/*!40111 SET SQL_NOTES=@OLD_SQL_NOTES */;
--- Dump completed on 2023-11-11 8:33:40
+-- Dump completed on 2023-11-12 15:56:47
diff --git a/sql/hpr.sql b/sql/hpr.sql
index 40a69a5..0a0318d 100644
--- a/sql/hpr.sql
+++ b/sql/hpr.sql
@@ -20485,7 +20485,11 @@ INSERT INTO `eps` (`id`, `date`, `title`, `duration`, `summary`, `notes`, `hosti
(3985,'2023-11-10','Bash snippet - be careful when feeding data to loops',1644,'A loop in a pipeline runs in a subshell','\nOverview
\nRecently Ken Fallon did a show on HPR, number\n3962, in which he used a Bash\npipeline of multiple commands feeding their output into a\nwhile
loop. In the loop he processed the lines produced by\nthe pipeline and used what he found to download audio files belonging to\na series with wget
.
\nThis was a great show and contained some excellent advice, but the\nuse of the format:
\npipeline | while read variable; do ...
\nreminded me of the \"gotcha\" I mentioned in my own show\n2699.
\nI thought it might be a good time to revisit this subject.
\nSo, what\'s the problem?
\nThe problem can be summarised as a side effect of pipelines.
\nWhat are pipelines?
\nPipelines are an amazingly useful feature of Bash (and other shells).\nThe general format is:
\ncommand1 | command2 ...
\nHere command1
runs in a subshell and produces output (on\nits standard output) which is connected via the pipe symbol\n(|
) to command2
where it becomes its\nstandard input. Many commands can be linked together in this\nway to achieve some powerful combined effects.
\nA very simple example of a pipeline might be:
\n$ printf 'World\nHello\n' | sort\nHello\nWorld
\nThe printf
command (≡\'command1\'
) writes two\nlines (separated by newlines) on standard output and this is\npassed to the sort
command\'s standard input\n(≡\'command2\'
) which then sorts these lines\nalphabetically.
\nCommands in the pipeline can be more complex than this, and in the\ncase we are discussing we can include a loop command such as\nwhile
.
\nFor example:
\n$ printf 'World\nHello\n' | sort | while read line; do echo "($line)"; done\n(Hello)\n(World)
\nHere, each line output by the sort
command is read into\nthe variable line
in the while
loop and is\nwritten out enclosed in parentheses.
\nNote that the loop is written on one line. The semi-colons are used\ninstead of the equivalent newlines.
\nVariables and subshells
\nWhat if the lines output by the loop need to be numbered?
\n$ i=0; printf 'World\nHello\n' | sort | while read line; do ((i++)); echo "$i) $line"; done\n1) Hello\n2) World
\nHere the variable \'i\'
is set to zero before the\npipeline. It could have been done on the line before of course. In the\nwhile
loop the variable is incremented on each iteration\nand included in the output.
\nYou might expect \'i\'
to be 2 once the loop exits but it\nis not. It will be zero in fact.
\nThe reason is that there are two \'i\'
variables. One is\ncreated when it\'s set to zero at the start before the pipeline. The\nother one is created in the loop as a \"clone\". The expression:
\n((i++))
\nboth creates the variable (where it is a copy of the one in the\nparent shell) and increments it.
\nWhen the subshell in which the loop runs completes, it will delete\nthis version of \'i\'
and the original one will simply\ncontain the zero that it was originally set to.
\nYou can see what happens in this slightly different example:
\n$ i=1; printf 'World\nHello\n' | sort | while read line; do ((i++)); echo "$i) $line"; done\n2) Hello\n3) World\n$ echo $i\n1
\nThese examples are fine, assuming the contents of variable\n\'i\'
incremented in the loop are not needed outside it.
\nThe thing to remember is that the same variable name used in a\nsubshell is a different variable; it is initialised with the value of\nthe \"parent\" variable but any changes are not passed back.
\nHow to avoid the\nloss of changes in the loop
\nTo solve this the loop needs to be run in the original shell, not a\nsubshell. The pipeline which is being read needs to be attached to the\nloop in a different way:
\n$ i=0; while read line; do ((i++)); echo "$i) $line"; done < <(printf 'World\nHello\n' | sort)\n1) Hello\n2) World\n$ echo $i\n2
\nWhat is being used here is process\nsubstitution. A list of commands or pipelines are enclosed with\nparentheses and a \'less than\'
sign prepended to the list\n(with no intervening spaces). This is functionally equivalent to a\n(temporary) file of data.
\nThe redirection feature allows for data being read from a\nfile in a loop. The general format of the command is:
\nwhile read variable\n do\n # Use the variable\n done < file
\nUsing process substitution instead of a file will achieve what is\nrequired if computations are being done in the loop and the results are\nwanted after it has finished.
\nBeware of this type of\nconstruct
\nThe following one-line command sequence looks similar to the version\nusing process substitution, but is just another form of pipeline:
\n$ i=0; while read line; do echo $line; ((i++)); done < /etc/passwd | head -n 5; echo $i\nroot:x:0:0:root:/root:/bin/bash\ndaemon:x:1:1:daemon:/usr/sbin:/usr/sbin/nologin\nbin:x:2:2:bin:/bin:/usr/sbin/nologin\nsys:x:3:3:sys:/dev:/usr/sbin/nologin\nsync:x:4:65534:sync:/bin:/bin/sync\n0
\nThis will display the first 5 lines of the file but does it by\nreading and writing the entire file and only showing the first 5 lines\nof what is written by the loop.
\nWhat is more, because the while
is in a subshell in a\npipeline changes to variable \'i\'
will be lost.
\nAdvice
\n\nUse the pipe-connected-to-loop layout if you\'re aware of\nthe pitfalls, but will not be affected by them.
\nUse the read-from-process-substitution format if you\nwant your loop to be complex and to read and write variables in the\nscript.
\nPersonally, I always use the second form in scripts, but if I\'m\nwriting a temporary one-line thing on the command line I usually use the\nfirst form.
\n
\nTracing pipelines (advanced)
\nI have always wondered about processes in Unix. The process you log\nin to, normally called a shell runs a command language\ninterpreter that executes commands read from the standard input or\nfrom a file. There are several such interpreters available, but we\'re\ndealing with bash here.
\nProcesses are fairly lightweight entities in Unix/Linux. They can be\ncreated and destroyed quickly, with minimal overhead. I used to work\nwith Digital Equipment Corporation\'s OpenVMS operating system\nwhich also uses processes - but these are much more expensive to create\nand destroy, and therefore slow and less readily used!
\nBash pipelines, as discussed, use subshells. The description\nin the Bash man page says:
\n\nEach command in a multi-command pipeline, where pipes are created, is\nexecuted in a subshell, which is a separate process.
\n
\nSo a subshell in this context is basically another child\nprocess of the main login process (or other parent process), running\nBash.
\nProcesses (subshells) can be created in other ways. One is to place a\ncollection of commands in parentheses. These can be simple Bash\ncommands, separated by semi-colons, or pipelines. For example:
\n$ (echo "World"; echo "Hello") | sort\nHello\nWorld
\nHere the strings \"World\"
and \"Hello\"
, each\nfollowed by a newline are created in a subshell and written to standard\noutput. These strings are piped to sort
and the end result\nis as shown.
\nNote that this is different from this example:
\n$ echo "World"; echo "Hello" | sort\nWorld\nHello
\nIn this case \"World\"
is written in a separate command,\nthen \"Hello\"
is written to a pipeline. All\nsort
sees is the output from the second echo
,\nwhich explains the output.
\nEach process has a unique numeric id value (the process id\nor PID). These can be seen with tools like ps
or\nhtop
. Each process holds its own PID in a Bash variable\ncalled BASHPID
.
\nKnowing all of this I decided to modify Ken\'s script from show\n3962 to show the processes being created - mainly for my interest,\nto get a better understanding of how Bash works. I am including it here\nin case it may be of interest to others.
\n#!/bin/bash\n\nseries_url="https://hackerpublicradio.org/hpr_mp3_rss.php?series=42&full=1&gomax=1"\ndownload_dir="./"\n\npidfile="/tmp/hpr3962.sh.out"\ncount=0\n\necho "Starting PID is $BASHPID" > $pidfile\n\n(echo "[1] $BASHPID" >> "$pidfile"; wget -q "${series_url}" -O -) |\\n (echo "[2] $BASHPID" >> "$pidfile"; xmlstarlet sel -T -t -m 'rss/channel/item' -v 'concat(enclosure/@url, "→", title)' -n -) |\\n (echo "[3] $BASHPID" >> "$pidfile"; sort) |\\n while read -r episode; do\n\n [ $count -le 1 ] && echo "[4] $BASHPID" >> "$pidfile"\n ((count++))\n\n url="$( echo "${episode}" | awk -F '→' '{print $1}' )"\n ext="$( basename "${url}" )"\n title="$( echo "${episode}" | awk -F '→' '{print $2}' | sed -e 's/[^A-Za-z0-9]/_/g' )"\n #wget "${url}" -O "${download_dir}/${title}.${ext}"\n done\n\necho "Final value of \$count = $count"\necho "Run 'cat $pidfile' to see the PID numbers"
\nThe point of doing this is to get information about the pipeline\nwhich feeds data into the while loop
. I kept the rest\nintact but commented out the wget
command.
\nFor each component of the pipeline I added an echo
\ncommand and enclosed it and the original command in parentheses, thus\nmaking a multi-command process. The echo
commands write a\nfixed number so you can tell which one is being executed, and it also\nwrites the contents of BASHPID
.
\nThe whole thing writes to a temporary file\n/tmp/hpr3962.sh.out
which can be examined once the script\nhas finished.
\nWhen the script is run it writes the following:
\n$ ./hpr3962.sh\nFinal value of $count = 0\nRun 'cat /tmp/hpr3962.sh.out' to see the PID numbers
\nThe file mentioned contains:
\nStarting PID is 80255\n[1] 80256\n[2] 80257\n[3] 80258\n[4] 80259\n[4] 80259
\nNote that the PID values are incremental. There is no guarantee that\nthis will be so. It will depend on whatever else the machine is\ndoing.
\nMessage number 4 is the same for every loop iteration, so I stopped\nit being written after two instances.
\nThe initial PID is the process running the script, not the login\n(parent) PID. You can see that each command in the pipeline runs in a\nseparate process (subshell), including the loop.
\nGiven that a standard pipeline generates a process per command, I was\nslightly surprised that the PID numbers were consecutive. It seems that\nBash optimises things so that only one process is run for each element\nof the pipe. I expect that it would be possible for more processes to be\ncreated by having pipelines within these parenthesised lists, but I\nhaven\'t tried it!
\nI found this test script quite revealing. I hope you find it useful\ntoo.
\nLinks
\n\n\n\n- Bash process substitution:\n
\n
\n\n- HPR shows referenced:\n
\n
\n\n',225,42,1,'CC-BY-SA','Bash,loop,process,shell',0,0,1),
(3992,'2023-11-21','Test recording on a wireless mic',223,'Archer72 tests out a wireless mic with a USB C receiver','LEKATO 2\nPack Wireless Microphone with Charging Case
\nhttps://www.amazon.com/gp/product/B0C4SNT6QK
\n\n- USB C
\n- Two microphones in a charging case
\n- Charge phone and use the receiver simultaneously
\n
\nClaims
\n\n- 75 ft. transmission range
\n- Wireless Mic can work continuously for 5 hours, and the charging box\ncan quickly charge the device 4 times. The total usage time reaches 25\nhours
\n
\nAxet Audio recorder on\nF-Droid
\nhttps://f-droid.org/packages/com.github.axet.audiorecorder
\n\n- Works to record stereo with this mic set
\n
\n',318,0,0,'CC-BY-SA','Recording, Microphone, Wireless, USB \'C\', F-droid, Android App',0,0,1),
(4221,'2024-10-07','HPR Community News for September 2024',0,'HPR Volunteers talk about shows released and comments posted in September 2024','',159,47,1,'CC-BY-SA','Community News',0,0,1),
-(4241,'2024-11-04','HPR Community News for October 2024',0,'HPR Volunteers talk about shows released and comments posted in October 2024','',159,47,1,'CC-BY-SA','Community News',0,0,1);
+(4241,'2024-11-04','HPR Community News for October 2024',0,'HPR Volunteers talk about shows released and comments posted in October 2024','',159,47,1,'CC-BY-SA','Community News',0,0,1),
+(3986,'2023-11-13','Optical media is not dead',435,'Archer72 shows command line options for creating and writing iso files','Brought up by Klaatu\non GnuWorldOrder.info
\n\n4.7Gb DVD - Actual capacity: 4.377Gb
\nWhat is the\nactual storage capacity of a dvd disc
\nA disc with a 25GB capacity is the equivalent of 23.28 gigabytes\nNormally rated at 50GB, in practice they can record about 46.57GB of\ndata
\nWhat is the\nmaximum capacity of a blu ray disc
\nGenerate ISO image from directory
\ngenisoimage -U -R -allow-lowercase -allow-multidot -o mydvd.iso "$1"
\nAlternative is mkisofs
\n-U\n\nAllows "untranslated" filenames, completely violating the ISO9660 standards described above. Enables the following flags: -d -l -N -allow-leading-dots -relaxed-filenames -allow-lowercase -allow-multidot -no-iso-translate. Allows more than one `.' character in the filename, as well as mixed-case filenames. This is useful on HP-UX, where the built-in cdfs filesystem does not recognize any extensions. Use with extreme caution.\n\n-R\n\nGenerate SUSP and RR records using the Rock Ridge protocol to further describe the files on the ISO9660 filesystem.\n\n[Wikipedia - ISO 9660](https://en.wikipedia.org/wiki/ISO_9660#SUSP "Wikipedia - ISO 9660")
\nCorrected command
\ngenisoimage -R -o mydvd.iso "$1"
\nBurning data to a DVD or Blu-ray
\nNote:
\nMake sure that the medium is not mounted when you begin to write to\nit. Mounting may happen automatically if the medium contains a readable\nfile system. In the best case, it will prevent the burn programs from\nusing the burner device. In the worst case, there will be misburns\nbecause read operations disturbed the drive. So if in doubt, do:
\numount /dev/sr0
\ngrowisofs has a small bug with blank BD-R media. It issues an error\nmessage after the burning is complete. Programs like k3b then believe\nthe whole burn run failed. To prevent this, either format the blank BD-R\nby dvd+rw-format /dev/sr0 before submitting it to growisofs or use\ngrowisofs option
\n-use-the-force-luke=spare:none
\nArchwiki - Optical disk\nburning
\nBurning an ISO image to\nCD, DVD, or BD
\nTo\nburn a readily prepared ISO image file isoimage.iso onto an optical\nmedium, run for CD:
\ncdrecord -v -sao dev=/dev/sr0 isoimage.iso
\nand for DVD or BD:
\ngrowisofs -dvd-compat -Z /dev/sr0=isoimage.iso
\nfor CD, DVD, BD:
\nxorriso -as cdrecord -v dev=/dev/sr0 -dao isoimage.iso
\nOther reading
\nArchwiki - Optical disk\ndrive
\nArchwiki - Optical disk\ndrive
\nDebian - Gensisoimage man\npage
\nDebian - Gensisoimage man\npage
\nDebian wiki -\ngenisoimage and xorrisofs
\nDebian wiki
\nArchiving data on Blu-ray\ndiscs
\nArchiving data
\nMount\nan ISO file and Burning it to CD-R/DVD-R/BluRay in Linux
\nMount an ISO file and\nBurning
\nWhat\nis the size capacity of my DVD, Dual Layer DVD or Blu-ray disc?
\nWhat is the size capacity of\nmy DVD
\nWikipedia ISO9660
\nWikipedia ISO9660
\n',318,0,0,'CC-BY-SA','Command line, Create ISO, Burn ISO, Optical media, DVD, CD, Blu-ray',0,0,0),
+(3987,'2023-11-14','The Grim Dawn',2391,'Sgoti rambles about a video game called Grim Dawn.','\nThis work is licensed under a Creative Commons\nAttribution-ShareAlike 4.0 International License.
\n',391,0,0,'CC-BY-SA','Grim Dawn, ARPG, Diablo 4, Video Games',0,0,0),
+(3988,'2023-11-15','Beeper.com',750,'operat0r talks about Beeper dot com a multi chat client','I talk about Beeper dot com a multi chat client
\nBeeper is a universal messaging app that lets you chat with anyone on\nany chat app, including Whatsapp, iMessage, Telegram and 12 other\nnetworks.
\nLinks
\n\n',36,0,1,'CC-BY-SA','chat,messaging,mobile',0,0,0),
+(3989,'2023-11-16','LastPass Security Update 1 November 2023',553,'LastPass was hacked, what should you do?','In 2022, LastPass disclosed that it had been hacked, and I think by\nnow just about everyone has heard about it. Now we have evidence that\npassword vaults have been hacked. So what does this mean, and what\nshould you do?
\nLinks:
\n\n',198,74,0,'CC-BY-SA','LastPass, password vault',0,0,0);
/*!40000 ALTER TABLE `eps` ENABLE KEYS */;
UNLOCK TABLES;
@@ -21409,4 +21413,4 @@ UNLOCK TABLES;
/*!40014 SET UNIQUE_CHECKS=@OLD_UNIQUE_CHECKS */;
/*!40111 SET SQL_NOTES=@OLD_SQL_NOTES */;
--- Dump completed on 2023-11-11 8:33:40
+-- Dump completed on 2023-11-12 15:56:47