mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-22 23:00:57 +08:00
773 lines
25 KiB
Markdown
773 lines
25 KiB
Markdown
[#]: collector: (lujun9972)
|
||
[#]: translator: (rakino)
|
||
[#]: reviewer: ( )
|
||
[#]: publisher: ( )
|
||
[#]: url: ( )
|
||
[#]: subject: (Run a command on binary files with this script)
|
||
[#]: via: (https://opensource.com/article/20/3/run-binaries-script)
|
||
[#]: author: (Nick Clifton https://opensource.com/users/nickclifton)
|
||
|
||
Run a command on binary files with this script
|
||
======
|
||
Try this simple script to easily run a command on binary files
|
||
regardless of their packaging.
|
||
![Binary code on a computer screen][1]
|
||
|
||
Examining files from the command-line is generally an easy thing to do. You just run the command you want, followed by a list of files to be examined. Dealing with binary files, however, is more complicated. These files are often packaged up into archives, tarballs, or other packaging formats. The run-on-binaries script provides a convenient way to run a command on a collection of files, regardless of how they are packaged.
|
||
|
||
The invocation of the script is quite simple:
|
||
|
||
|
||
```
|
||
`run-on-binaries <file(s}>`
|
||
```
|
||
|
||
So, for example:
|
||
|
||
|
||
```
|
||
`run-on-binaries /usr/bin/ls foo.rpm`
|
||
```
|
||
|
||
will list all of the files inside the **foo.rpm** file, while:
|
||
|
||
|
||
```
|
||
`run-on-binaries /usr/bin/readelf -a libc.a`
|
||
```
|
||
|
||
will run the **readelf** program, with the **-a** command-line option, on all of the object files inside the **libc.a library**.
|
||
|
||
If necessary, the script can be passed a file containing a list of other files to be processed, rather than specifying them on the command line—like this:
|
||
|
||
|
||
```
|
||
`run-on-binaries --files-from=foo.lst /usr/bin/ps2ascii`
|
||
```
|
||
|
||
This will run the **ps2ascii** script on all of the files listed in **foo.lst**. (The files just need to be separated by white space. There can be multiple files on a single line if desired).
|
||
|
||
Also, a skip list can be provided to stop the script from processing specified files:
|
||
|
||
|
||
```
|
||
`run-on-binaries --skip-list=skip.lst /usr/bin/wc *`
|
||
```
|
||
|
||
This will run the **wc** program on all of the files in the current directory, except for those specified in **skip.lst**.
|
||
|
||
The script does not recurse into directories, but this can be handled by combining it with the **find** command, like this:
|
||
|
||
|
||
```
|
||
`find . -type f -exec run-on-binaries @ ;`
|
||
```
|
||
|
||
or
|
||
|
||
|
||
```
|
||
`find . -type d -exec run-on-binaries @/* ;`
|
||
```
|
||
|
||
The only difference between these two invocations is that the second one only runs the target program once per directory, but gives it a long command-line of all of the files in the directory.
|
||
|
||
Though convenient, the script is lacking in several areas. Right now, it does not examine the PATH environment variable to find the command that it is asked to run, so a full path must be provided. Also, the script ought to be able to handle recursion on its own, without needing help from the find command.
|
||
|
||
The run-on-binaries script is part of the annobin package, which is available on Fedora. The sources for annobin can also be obtained from the git repository at <git://sourceware.org/git/annobin.git>.
|
||
|
||
### The script
|
||
|
||
|
||
```
|
||
#!/bin/bash
|
||
|
||
# Script to run another script/program on the executables inside a given file.
|
||
#
|
||
# Created by Nick Clifton. <[nickc@redhat.com][2]>
|
||
# Copyright (c) 2018 Red Hat.
|
||
#
|
||
# This is free software; you can redistribute it and/or modify it
|
||
# under the terms of the GNU General Public License as published
|
||
# by the Free Software Foundation; either version 3, or (at your
|
||
# option) any later version.
|
||
|
||
# It is distributed in the hope that it will be useful, but
|
||
# WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
# GNU General Public License for more details.
|
||
#
|
||
# Usage:
|
||
# run-on-binaries-in [options] program [options-for-the-program] file(s)
|
||
#
|
||
# This script does not handle directories. This is deliberate.
|
||
# It is intended that if recursion is needed then it will be
|
||
# invoked from find, like this:
|
||
#
|
||
# find . -name "*.rpm" -exec run-on-binaries-in <script-to-run> {} \;
|
||
|
||
version=1.0
|
||
|
||
help ()
|
||
{
|
||
# The following exec goop is so that we don't have to manually
|
||
# redirect every message to stderr in this function.
|
||
exec 4>&1 # save stdout fd to fd #4
|
||
exec 1>&2 # redirect stdout to stderr
|
||
|
||
cat <<__EOM__
|
||
|
||
This is a shell script to run another script/program on one or more binary
|
||
files. If the file(s) specified are archives of some kind (including rpms)
|
||
then the script/program is run on the binary executables inside the archive.
|
||
|
||
Usage: $prog {options} program {options-for-the-program} files(s)
|
||
|
||
{options} are:
|
||
-h --help Display this information and then exit.
|
||
-v --version Report the version number of this script.
|
||
-V --verbose Report on progress.
|
||
-q --quiet Do not include the script name in the output.
|
||
-i --ignore Silently ignore files that are not executables or archives.
|
||
-p=<TEXT> --prefix=<TEXT> Prefix normal output with this string.
|
||
-t=<DIR> --tmpdir=<DIR> Temporary directory to use when opening archives.
|
||
-f=<FILE> --files-from=<FILE> Process files listed in <FILE>.
|
||
-s=<FILE> --skip-list=<FILE> Skip any file listed in <FILE>.
|
||
-- Stop accumulating options.
|
||
|
||
Examples:
|
||
|
||
$prog hardened foo.rpm
|
||
Runs the hardened script on the executable
|
||
files inside foo.rpm.
|
||
|
||
$prog check-abi -v fred.tar.xz
|
||
Runs the check-abi script on the decompressed
|
||
contents of the fred.tar.xz archive, passing the
|
||
-v option to check-abi as it does so.
|
||
|
||
$prog -V -f=list.txt readelf -a
|
||
Runs the readelf program, with the -a option on
|
||
every file listed in the list.txt. Describes
|
||
what is being done as it works.
|
||
|
||
$prog -v -- -fred -a jim -b bert -- -c harry
|
||
Runs the script "-fred" on the files jim, bert,
|
||
"-c" and harry. Passes the options "-a" and
|
||
"-b" to the script (even when run on jim).
|
||
Reports the version of this script as well.
|
||
|
||
__EOM__
|
||
exec 1>&4 # Copy stdout fd back from temporary save fd, #4
|
||
}
|
||
|
||
main ()
|
||
{
|
||
init
|
||
|
||
parse_args ${1+"$@"}
|
||
|
||
if [ $failed -eq 0 ];
|
||
then
|
||
run_script_on_files
|
||
fi
|
||
|
||
if [ $failed -ne 0 ];
|
||
then
|
||
exit 1
|
||
else
|
||
exit 0
|
||
fi
|
||
}
|
||
|
||
report ()
|
||
{
|
||
if [ $quiet -eq 0 ];
|
||
then
|
||
echo -n $prog": "
|
||
fi
|
||
|
||
echo ${1+"$@"}
|
||
}
|
||
|
||
ice ()
|
||
{
|
||
report "Internal error: " ${1+"$@"}
|
||
exit 1
|
||
}
|
||
|
||
fail ()
|
||
{
|
||
report "Failure:" ${1+"$@"}
|
||
failed=1
|
||
}
|
||
|
||
verbose ()
|
||
{
|
||
if [ $verbose -ne 0 ]
|
||
then
|
||
report ${1+"$@"}
|
||
fi
|
||
}
|
||
|
||
# Initialise global variables.
|
||
init ()
|
||
{
|
||
files[0]="";
|
||
# num_files is the number of files to be scanned.
|
||
# files[0] is the script to run on the files.
|
||
num_files=0;
|
||
|
||
script=""
|
||
script_opts="";
|
||
|
||
prog_opts="-i"
|
||
|
||
tmpdir=/dev/shm
|
||
prefix=""
|
||
files_from=""
|
||
skip_list=""
|
||
|
||
failed=0
|
||
verbose=0
|
||
ignore=0
|
||
quiet=0
|
||
}
|
||
|
||
# Parse our command line
|
||
parse_args ()
|
||
{
|
||
abs_prog=$0;
|
||
prog=`basename $abs_prog`;
|
||
|
||
# Locate any additional command line switches
|
||
# Likewise accumulate non-switches to the files list.
|
||
while [ $# -gt 0 ]
|
||
do
|
||
optname="`echo $1 | sed 's,=.*,,'`"
|
||
optarg="`echo $1 | sed 's,^[^=]*=,,'`"
|
||
case "$optname" in
|
||
-v | --version)
|
||
report "version: $version"
|
||
;;
|
||
-h | --help)
|
||
help
|
||
exit 0
|
||
;;
|
||
-q | --quiet)
|
||
quiet=1;
|
||
prog_opts="$prog_opts -q"
|
||
;;
|
||
-V | --verbose)
|
||
if [ $verbose -eq 1 ];
|
||
then
|
||
# This has the effect of cancelling out the prog_opts="-i"
|
||
# in the init function, so that recursive invocations of this
|
||
# script will complain about unrecognised file types.
|
||
if [ $quiet -eq 0 ];
|
||
then
|
||
prog_opts="-V -V"
|
||
else
|
||
prog_opts="-V -V -q"
|
||
fi
|
||
else
|
||
verbose=1;
|
||
prog_opts="$prog_opts -V"
|
||
fi
|
||
;;
|
||
-i | --ignore)
|
||
ignore=1
|
||
;;
|
||
-t | --tmpdir)
|
||
if test "x$optarg" = "x$optname" ;
|
||
then
|
||
shift
|
||
if [ $# -eq 0 ]
|
||
then
|
||
fail "$optname needs a directory name"
|
||
else
|
||
tmpdir=$1
|
||
fi
|
||
else
|
||
tmpdir="$optarg"
|
||
fi
|
||
;;
|
||
-p | --prefix)
|
||
if test "x$optarg" = "x$optname" ;
|
||
then
|
||
shift
|
||
if [ $# -eq 0 ]
|
||
then
|
||
fail "$optname needs a string argument"
|
||
else
|
||
prefix=$1
|
||
fi
|
||
else
|
||
prefix="$optarg"
|
||
fi
|
||
;;
|
||
-f | --files_from)
|
||
if test "x$optarg" = "x$optname" ;
|
||
then
|
||
shift
|
||
if [ $# -eq 0 ]
|
||
then
|
||
fail "$optname needs a file name"
|
||
else
|
||
files_from=$1
|
||
fi
|
||
else
|
||
files_from="$optarg"
|
||
fi
|
||
;;
|
||
|
||
-s | --skip-list)
|
||
if test "x$optarg" = "x$optname" ;
|
||
then
|
||
shift
|
||
if [ $# -eq 0 ]
|
||
then
|
||
fail "$optname needs a file name"
|
||
else
|
||
skip_list=$1
|
||
fi
|
||
else
|
||
skip_list="$optarg"
|
||
fi
|
||
;;
|
||
|
||
--)
|
||
shift
|
||
break;
|
||
;;
|
||
--*)
|
||
fail "unrecognised option: $1"
|
||
help
|
||
;;
|
||
*)
|
||
script="$1";
|
||
if ! [ -a "$script" ]
|
||
then
|
||
fail "$script: program/script not found"
|
||
elif ! [ -x "$script" ]
|
||
then
|
||
fail "$script: program/script not executable"
|
||
fi
|
||
# After we have seen the first non-option we stop
|
||
# accumulating options for this script and instead
|
||
# start accumulating options for the script to be
|
||
# run.
|
||
shift
|
||
break;
|
||
;;
|
||
esac
|
||
shift
|
||
done
|
||
|
||
# Read in the contents of the --file-from list, if specified.
|
||
if test "x$files_from" != "x" ;
|
||
then
|
||
if ! [ -a "$files_from" ]
|
||
then
|
||
fail "$files_from: file not found"
|
||
elif ! [ -r "$files_from" ]
|
||
then
|
||
fail "$files_from: file not readable"
|
||
else
|
||
eval 'files=($(cat $files_from))'
|
||
num_files=${#files[*]}
|
||
fi
|
||
fi
|
||
skip_files[foo]=bar
|
||
|
||
# Check that the skip list exists, if specified.
|
||
if test "x$skip_list" != "x" ;
|
||
then
|
||
if ! [ -a "$skip_list" ]
|
||
then
|
||
fail "$skip_list: file not found"
|
||
elif ! [ -r "$skip_list" ]
|
||
then
|
||
fail "$files_from: file not readable"
|
||
fi
|
||
fi
|
||
|
||
# Accumulate any remaining arguments separating out the arguments
|
||
# for the script from the names of the files to scan.
|
||
while [ $# -gt 0 ]
|
||
do
|
||
optname="`echo $1 | sed 's,=.*,,'`"
|
||
optarg="`echo $1 | sed 's,^[^=]*=,,'`"
|
||
case "$optname" in
|
||
--)
|
||
shift
|
||
break;
|
||
;;
|
||
-*)
|
||
script_opts="$script_opts $1"
|
||
;;
|
||
*)
|
||
files[$num_files]="$1";
|
||
let "num_files++"
|
||
;;
|
||
esac
|
||
shift
|
||
done
|
||
|
||
# Accumulate any remaining arguments without processing them.
|
||
while [ $# -gt 0 ]
|
||
do
|
||
files[$num_files]="$1";
|
||
let "num_files++";
|
||
shift
|
||
done
|
||
|
||
if [ $num_files -gt 0 ];
|
||
then
|
||
# Remember that we are counting from zero not one.
|
||
let "num_files--"
|
||
else
|
||
fail "Must specify a program/script and at least one file to scan."
|
||
fi
|
||
}
|
||
|
||
run_script_on_files ()
|
||
{
|
||
local i
|
||
|
||
i=0;
|
||
while [ $i -le $num_files ]
|
||
do
|
||
run_on_file i
|
||
let "i++"
|
||
done
|
||
}
|
||
|
||
# syntax: run <command> [<args>]
|
||
# If being verbose report the command being run, and
|
||
# the directory in which it is run.
|
||
run ()
|
||
{
|
||
local where
|
||
|
||
if test "x$1" = "x" ;
|
||
then
|
||
fail "run() called without an argument."
|
||
fi
|
||
|
||
verbose " Running: ${1+$@}"
|
||
|
||
${1+$@}
|
||
}
|
||
|
||
decompress ()
|
||
{
|
||
local abs_file decompressor decomp_args orig_file base_file
|
||
|
||
# Paranoia checks - the user should never encounter these.
|
||
if test "x$4" = "x" ;
|
||
then
|
||
ice "decompress called with too few arguments"
|
||
fi
|
||
if test "x$5" != "x" ;
|
||
then
|
||
ice "decompress called with too many arguments"
|
||
fi
|
||
|
||
abs_file=$1
|
||
decompressor=$2
|
||
decomp_args=$3
|
||
orig_file=$4
|
||
|
||
base_file=`basename $abs_file`
|
||
|
||
run cp $abs_file $base_file
|
||
run $decompressor $decomp_args $base_file
|
||
if [ $? != 0 ];
|
||
then
|
||
fail "$orig_file: Unable to decompress"
|
||
fi
|
||
|
||
rm -f $base_file
|
||
}
|
||
|
||
run_on_file ()
|
||
{
|
||
local file
|
||
|
||
# Paranoia checks - the user should never encounter these.
|
||
if test "x$1" = "x" ;
|
||
then
|
||
ice "scan_file called without an argument"
|
||
fi
|
||
if test "x$2" != "x" ;
|
||
then
|
||
ice "scan_file called with too many arguments"
|
||
fi
|
||
|
||
# Use quotes when accessing files in order to preserve
|
||
# any spaces that might be in the directory name.
|
||
file="${files[$1]}";
|
||
|
||
# Catch names that start with a dash - they might confuse readelf
|
||
if test "x${file:0:1}" = "x-" ;
|
||
then
|
||
file="./$file"
|
||
fi
|
||
|
||
# See if we should skip this file.
|
||
if test "x$skip_list" != "x" ;
|
||
then
|
||
# This regexp looks for $file being the first text on a line, either
|
||
# on its own, or with additional text separated from it by at least
|
||
# one space character. So searching for "fred" in the following gives:
|
||
# fr <\- no match
|
||
# fred <\- match
|
||
# fredjim <\- no match
|
||
# fred bert <\- match
|
||
regexp="^$file[^[:graph:]]*"
|
||
grep --silent --regexp="$regexp" $skip_list
|
||
if [ $? = 0 ];
|
||
then
|
||
verbose "$file: skipping"
|
||
return
|
||
fi
|
||
fi
|
||
|
||
# Check the file.
|
||
if ! [ -a "$file" ]
|
||
then
|
||
fail "$file: file not found"
|
||
return
|
||
elif ! [ -r "$file" ]
|
||
then
|
||
if [ $ignore -eq 0 ];
|
||
then
|
||
fail "$file: not readable"
|
||
fi
|
||
return
|
||
elif [ -d "$file" ]
|
||
then
|
||
if [ $ignore -eq 0 ];
|
||
then
|
||
if [ $num_files -gt 1 ];
|
||
then
|
||
verbose "$file: skipping - it is a directory"
|
||
else
|
||
report "$file: skipping - it is a directory"
|
||
fi
|
||
fi
|
||
return
|
||
elif ! [ -f "$file" ]
|
||
then
|
||
if [ $ignore -eq 0 ];
|
||
then
|
||
fail "$file: not an ordinary file"
|
||
fi
|
||
return
|
||
fi
|
||
|
||
file_type=`file -b $file`
|
||
case "$file_type" in
|
||
*"ELF "*)
|
||
verbose "$file: ELF format - running script/program"
|
||
if test "x$prefix" != "x" ;
|
||
then
|
||
report "$prefix:"
|
||
fi
|
||
run $script $script_opts $file
|
||
return
|
||
;;
|
||
"RPM "*)
|
||
verbose "$file: RPM format."
|
||
;;
|
||
*" cpio "*)
|
||
verbose "$file: CPIO format."
|
||
;;
|
||
*"tar "*)
|
||
verbose "$file: TAR archive."
|
||
;;
|
||
*"Zip archive"*)
|
||
verbose "$file: ZIP archive."
|
||
;;
|
||
*"ar archive"*)
|
||
verbose "$file: AR archive."
|
||
;;
|
||
*"bzip2 compressed data"*)
|
||
verbose "$file: contains bzip2 compressed data"
|
||
;;
|
||
*"gzip compressed data"*)
|
||
verbose "$file: contains gzip compressed data"
|
||
;;
|
||
*"lzip compressed data"*)
|
||
verbose "$file: contains lzip compressed data"
|
||
;;
|
||
*"XZ compressed data"*)
|
||
verbose "$file: contains xz compressed data"
|
||
;;
|
||
*"shell script"* | *"ASCII text"*)
|
||
if [ $ignore -eq 0 ];
|
||
then
|
||
fail "$file: test/scripts cannot be scanned."
|
||
fi
|
||
return
|
||
;;
|
||
*"symbolic link"*)
|
||
if [ $ignore -eq 0 ];
|
||
then
|
||
# FIXME: We ought to be able to follow symbolic links
|
||
fail "$file: symbolic links are not followed."
|
||
fi
|
||
return
|
||
;;
|
||
*)
|
||
if [ $ignore -eq 0 ];
|
||
then
|
||
fail "$file: Unsupported file type: $file_type"
|
||
fi
|
||
return
|
||
;;
|
||
esac
|
||
|
||
# We now know that we will need a temporary directory
|
||
# so create one, and create paths to the file and scripts.
|
||
if test "x${file:0:1}" = "x/" ;
|
||
then
|
||
abs_file=$file
|
||
else
|
||
abs_file="$PWD/$file"
|
||
fi
|
||
|
||
if test "x${abs_prog:0:1}" != "x/" ;
|
||
then
|
||
abs_prog="$PWD/$abs_prog"
|
||
fi
|
||
|
||
if test "x${script:0:1}" = "x/" ;
|
||
then
|
||
abs_script=$script
|
||
else
|
||
abs_script="$PWD/$script"
|
||
fi
|
||
|
||
tmp_root=$tmpdir/delme.run.on.binary
|
||
run mkdir -p "$tmp_root/$file"
|
||
|
||
verbose " Changing to directory: $tmp_root/$file"
|
||
pushd "$tmp_root/$file" > /dev/null
|
||
if [ $? != 0 ];
|
||
then
|
||
fail "Unable to change to temporary directory: $tmp_root/$file"
|
||
return
|
||
fi
|
||
|
||
# Run the file type switch again, although this time we do not need to
|
||
# check for unrecognised types. (But we do, just in case...)
|
||
# Note since are transforming the file we re-invoke the run-on-binaries
|
||
# script on the decoded contents. This allows for archives that contain
|
||
# other archives, and so on. We normally pass the -i option to the
|
||
# invoked script so that it will not complain about unrecognised files in
|
||
# the decoded archive, although we do not do this when running in very
|
||
# verbose mode. We also pass an extended -t option to ensure that any
|
||
# sub-archives are extracted into a unique directory tree.
|
||
|
||
case "$file_type" in
|
||
"RPM "*)
|
||
# The output redirect confuses the run function...
|
||
verbose " Running: rpm2cpio $abs_file > delme.cpio"
|
||
rpm2cpio $abs_file > delme.cpio
|
||
if [ $? != 0 ];
|
||
then
|
||
fail "$file: Unable to extract from rpm archive"
|
||
else
|
||
# Save time - run cpio now.
|
||
run cpio --quiet --extract --make-directories --file delme.cpio
|
||
if [ $? != 0 ];
|
||
then
|
||
fail "$file: Unable to extract files from cpio archive"
|
||
fi
|
||
run rm -f delme.cpio
|
||
fi
|
||
;;
|
||
|
||
*" cpio "*)
|
||
run cpio --quiet --extract --make-directories --file=$abs_file
|
||
if [ $? != 0 ];
|
||
then
|
||
fail "$file: Unable to extract files from cpio archive"
|
||
fi
|
||
;;
|
||
|
||
*"tar "*)
|
||
run tar --extract --file=$abs_file
|
||
if [ $? != 0 ];
|
||
then
|
||
fail "$file: Unable to extract files from tarball"
|
||
fi
|
||
;;
|
||
|
||
*"ar archive"*)
|
||
run ar x $abs_file
|
||
if [ $? != 0 ];
|
||
then
|
||
fail "$file: Unable to extract files from ar archive"
|
||
fi
|
||
;;
|
||
|
||
*"Zip archive"*)
|
||
decompress $abs_file unzip "-q" $file
|
||
;;
|
||
*"bzip2 compressed data"*)
|
||
decompress $abs_file bzip2 "--quiet --decompress" $file
|
||
;;
|
||
*"gzip compressed data"*)
|
||
decompress $abs_file gzip "--quiet --decompress" $file
|
||
;;
|
||
*"lzip compressed data"*)
|
||
decompress $abs_file lzip "--quiet --decompress" $file
|
||
;;
|
||
*"XZ compressed data"*)
|
||
decompress $abs_file xz "--quiet --decompress" $file
|
||
;;
|
||
*)
|
||
ice "unhandled file type: $file_type"
|
||
;;
|
||
esac
|
||
|
||
if [ $failed -eq 0 ];
|
||
then
|
||
# Now scan the file(s) created in the previous step.
|
||
run find . -type f -execdir $abs_prog $prog_opts -t=$tmp_root/$file -p=$file $abs_script $script_opts {} +
|
||
fi
|
||
|
||
verbose " Deleting temporary directory: $tmp_root"
|
||
rm -fr $tmp_root
|
||
|
||
verbose " Return to previous directory"
|
||
popd > /dev/null
|
||
}
|
||
|
||
# Invoke main
|
||
main ${1+"$@"}
|
||
```
|
||
|
||
|
||
|
||
Git has extensions for handling binary blobs such as multimedia files, so today we will learn how...
|
||
|
||
--------------------------------------------------------------------------------
|
||
|
||
via: https://opensource.com/article/20/3/run-binaries-script
|
||
|
||
作者:[Nick Clifton][a]
|
||
选题:[lujun9972][b]
|
||
译者:[rakino](https://github.com/rakino)
|
||
校对:[校对者ID](https://github.com/校对者ID)
|
||
|
||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||
|
||
[a]: https://opensource.com/users/nickclifton
|
||
[b]: https://github.com/lujun9972
|
||
[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/binary_code_computer_screen.png?itok=7IzHK1nn (Binary code on a computer screen)
|
||
[2]: mailto:nickc@redhat.com
|