<?xml version="1.0" encoding="utf-8"?>
<?xml-stylesheet type="text/xsl" href="../assets/xml/rss.xsl" media="all"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:atom="http://www.w3.org/2005/Atom"><channel><title>Компы и автомобили (Posts about Скрипты)</title><link>https://at-hacker.in/</link><description></description><atom:link href="https://at-hacker.in/categories/skripty.xml" rel="self" type="application/rss+xml"></atom:link><language>en</language><copyright>Contents © 2026 &lt;a href="mailto:"&gt;at-hacker&lt;/a&gt; </copyright><lastBuildDate>Sat, 16 May 2026 20:41:20 GMT</lastBuildDate><generator>Nikola (getnikola.com)</generator><docs>http://blogs.law.harvard.edu/tech/rss</docs><item><title>Как рулить инсталляциями server core</title><link>https://at-hacker.in/posts/kak-rulit-installyaciyami-server-core/</link><dc:creator>at-hacker</dc:creator><description>&lt;p&gt;Нашёл мега-полезную &lt;a href="http://blogs.technet.com/b/bruce_adamczak/archive/2013/01/15/2012-core-survival-guide.aspx"&gt;ссылку&lt;/a&gt; про то, как можно жить с server core. Прямо так и называется: «Сore 2012: Руководство по выживанию». Много всяких полезных штук — как настраивать сетку, как просматривать логи и так далее, через powershell. про то, как можно жить с server core. Прямо так и называется: «Сore 2012: Руководство по выживанию». Много всяких полезных штук — как настраивать сетку, как просматривать логи и так далее, через powershell.&lt;/p&gt;</description><category>Windows</category><category>Администрирование</category><category>Скрипты</category><guid>https://at-hacker.in/posts/kak-rulit-installyaciyami-server-core/</guid><pubDate>Mon, 02 Dec 2013 13:18:00 GMT</pubDate></item><item><title>Удалённое выполнение команд по ssh</title><link>https://at-hacker.in/posts/udalyonnoe-vypolnenie-komand-po-ssh/</link><dc:creator>at-hacker</dc:creator><description>&lt;p&gt;В свете полученных знаний из ссылок в предыдущем посте решил начать жить по-новому. А именно — бэкапные файлы с фри на фрю складывать не по SMB или FTP, как раньше, а по scp или ssh. Подумал, и решил, что удобнее будет через ssh — туда загоняется поток через pipe, а на удалённой стороне из pipe складывается в локальнный (для удалённого сервера-хранилища бэкапов) файл. Проверил даже скорость — в конкретной сети гиговый файл через SMB скопировался за  1 минуту 48 секунд, а то же самое, но через ssh — за 1 минуту 35 секунд. Профит налицо :-)&lt;/p&gt;
&lt;p&gt;Запустил mcedit и начал ваять. Со скриптом бэкапа системы проблем не возникло:&lt;/p&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;&lt;span class="c1"&gt;# cat system_backup.sh&lt;/span&gt;
&lt;span class="c1"&gt;#!/bin/sh&lt;/span&gt;
&lt;span class="n"&gt;backup_dir&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="s2"&gt;"/home/backup_operator/backups/os/buhserver-ttr"&lt;/span&gt;
&lt;span class="n"&gt;keyfile&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="s2"&gt;"/home/gateway/service/keyfile.ppk"&lt;/span&gt;
&lt;span class="n"&gt;username&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="s2"&gt;"backup_operator@backup-ttr"&lt;/span&gt;

&lt;span class="n"&gt;DATE&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="err"&gt;`&lt;/span&gt;&lt;span class="n"&gt;date&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;+%&lt;/span&gt;&lt;span class="n"&gt;Y&lt;/span&gt;&lt;span class="o"&gt;.%&lt;/span&gt;&lt;span class="n"&gt;m&lt;/span&gt;&lt;span class="o"&gt;.%&lt;/span&gt;&lt;span class="n"&gt;d&lt;/span&gt;&lt;span class="err"&gt;`&lt;/span&gt;

&lt;span class="n"&gt;dump&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;0&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;L&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;u&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;C&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="mi"&gt;32&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;f&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="err"&gt;—&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;/&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;|&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="n"&gt;gzip&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;5&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;|&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="n"&gt;ssh&lt;/span&gt;&lt;span class="w"&gt;  &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;q&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;i&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;$&lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="n"&gt;keyfile&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;$&lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="n"&gt;username&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"cat &amp;gt; ${backup_dir}/${DATE}_root.dump.gz"&lt;/span&gt;
&lt;span class="n"&gt;dump&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;0&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;L&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;u&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;C&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="mi"&gt;32&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;f&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="err"&gt;—&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;/&lt;/span&gt;&lt;span class="n"&gt;usr&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;|&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="n"&gt;gzip&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;5&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;|&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="n"&gt;ssh&lt;/span&gt;&lt;span class="w"&gt;  &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;q&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;i&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;$&lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="n"&gt;keyfile&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;$&lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="n"&gt;username&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"cat &amp;gt; ${backup_dir}/${DATE}_usr.dump.gz"&lt;/span&gt;
&lt;span class="n"&gt;dump&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;0&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;L&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;u&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;C&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="mi"&gt;32&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;f&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="err"&gt;—&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;/&lt;/span&gt;&lt;span class="k"&gt;var&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;|&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="n"&gt;gzip&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;5&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;|&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="n"&gt;ssh&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;q&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;i&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;$&lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="n"&gt;keyfile&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;$&lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="n"&gt;username&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"cat &amp;gt; ${backup_dir}/${DATE}_var.dump.gz"&lt;/span&gt;
&lt;span class="n"&gt;dump&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;0&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;L&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;u&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;C&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="mi"&gt;32&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;f&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="err"&gt;—&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;/&lt;/span&gt;&lt;span class="n"&gt;home&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;|&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="n"&gt;gzip&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;5&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;|&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="n"&gt;ssh&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;q&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="n"&gt;i&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;$&lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="n"&gt;keyfile&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;$&lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="n"&gt;username&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"cat &amp;gt; ${backup_dir}/${DATE}_home.dump."&lt;/span&gt;
&lt;/pre&gt;&lt;/div&gt;

&lt;p&gt;keyfile — это private key для авторизации на удалённом сервере, backup_operator — имя пользователя на том сервере, а backup-ttr — имя самого сервера, где у нас бэкапы хранятся. Получается следующее — dump дампит раздел в stdout, откуда это перегоняется в gzip через pipe, откуда перегоняется вот в эту конструкцию:&lt;/p&gt;
&lt;p&gt;ssh -q -i ${keyfile} ${username} "cat &amp;gt; ${backup_dir}/${DATE}_root.dump.gz"
Т. е. мы подключаемся к серверу backup-ttr, и выполняем там команду "&lt;em&gt;cat &amp;gt; {backup_dir}/${DATE}_root.dump.gz&lt;/em&gt;". Которая читает из stdin сжатый gzip'ом и прокинутый по сети поток и сохраняет его в файл {backup_dir}/${DATE}_root.dump.gz.". Которая читает из stdin сжатый gzip'ом и прокинутый по сети поток и сохраняет его в файл {backup_dir}/${DATE}_root.dump.gz.&lt;/p&gt;
&lt;p&gt;В общем, тут всё просто. А вот с бэкапом почты пришлось повозиться. Было принято решение не просто каждый день сваливать полный бэкап почтовой базы и удалять старые копии, а сделать инкрементный бэкап. А для этого нужно производить определённые манипуляции на стороне хранилища бэкапов. Можно, конечно, сделать второй скрипт, и запускать его по крону на стороне хранилища. Но «Шурик, это же не наш метод!». Это придётся контролировать две точки выполнения одной операции — бэкапа почты, так как задействованы будут два скрипта, два крона, два компа и т. д.&lt;/p&gt;
&lt;p&gt;Суть задачи такова. Допустим, у нас в день 1 выполняется полный бэкап почтовой базы. В день 2 — только то, что изменилось или добавилось со времени исполнения 1-го бэкапа, отработавшего в день 1. В день 3 — только изменения с момента бэкапа в день 2, и так далее до конца недели. Допустим, за первую неделю у нас сделано 7 архивов, с именами, соответственно, mail1.tgz, mail2.tgz, ... mail7.tgz. Где mail1.tgz — это полная копия базы, а остальные — дельты изменений. Неделя закончилась, надо как-то сделать ротацию бэкапов. То есть, куда-то деть эти 7 архивов. Можно их тупо удалить. Но как-то хочется, чтобы у нас хотя бы двухнедельная история бэкапов хранилась. Самое простое — эти архивы переименовать. Например, по схеме mail1.tgz.bak, mail2.tgz.bak и так далее. Тут-то и началось самое интересное.&lt;/p&gt;
&lt;p&gt;Сначала решил сделать как-нибудь так:&lt;/p&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;for&lt;span class="w"&gt; &lt;/span&gt;i&lt;span class="w"&gt; &lt;/span&gt;in&lt;span class="w"&gt; &lt;/span&gt;`ssh&lt;span class="w"&gt; &lt;/span&gt;-q&lt;span class="w"&gt; &lt;/span&gt;-i&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="cp"&gt;${&lt;/span&gt;&lt;span class="n"&gt;keyfile&lt;/span&gt;&lt;span class="cp"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="cp"&gt;${&lt;/span&gt;&lt;span class="n"&gt;username&lt;/span&gt;&lt;span class="cp"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;"ls&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="cp"&gt;${&lt;/span&gt;&lt;span class="n"&gt;backup_dir&lt;/span&gt;&lt;span class="cp"&gt;}&lt;/span&gt;/*.gz"`
do
&lt;span class="w"&gt;    &lt;/span&gt;ssh&lt;span class="w"&gt; &lt;/span&gt;-q&lt;span class="w"&gt; &lt;/span&gt;-i&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="cp"&gt;${&lt;/span&gt;&lt;span class="n"&gt;keyfile&lt;/span&gt;&lt;span class="cp"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="cp"&gt;${&lt;/span&gt;&lt;span class="n"&gt;username&lt;/span&gt;&lt;span class="cp"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;"mv&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="cp"&gt;${&lt;/span&gt;&lt;span class="n"&gt;i&lt;/span&gt;&lt;span class="cp"&gt;}&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="cp"&gt;${&lt;/span&gt;&lt;span class="n"&gt;i&lt;/span&gt;&lt;span class="cp"&gt;}&lt;/span&gt;.bak"
done
&lt;/pre&gt;&lt;/div&gt;

&lt;p&gt;Но это получается 1 вызов ssh для получения списка файлов, и потом ещё в цикле 7 раз запускается ssh, чтобы переименовать каждый отдельный файл. Касичок кагбэ, неоптимальненко.&lt;/p&gt;
&lt;p&gt;Камрады в интернете посоветовали сделать как-нибудь так:&lt;/p&gt;
&lt;p&gt;ssh -q user@remoteserver 'ls -f /path/*.tgz | while read f; do "$"» };done'
Я попробовал, не прокатило. Полез искать точный синтаксис однострочного оформления циклов в shell. Нашёл несколько примеров, подставил, но ни один не прокатил. Я так понял, что приведённые примеры — в основном для линуксового bash, а у меня фрёвая /bin/sh, поэтому и не прокатывает. Потом нашёл обрывочные сведения, что циклы типа for, foreach, while — «is not single-line friendly». Но коллективный разум мне сурово возразил, что я не прав.&lt;/p&gt;
&lt;p&gt;Копаясь дальше, я обнаружил, что реально, однострочные конструкции типа&lt;/p&gt;
&lt;p&gt;ls | while read f;do echo ${f};done
работают, будучи запущены в /bin/sh и локально. Но не работают удалённо. Потом до меня дошло, что локально-то я запустил /bin/sh, и в ней пускаю все эти конструкции. Но по дефолту у пользователя backup_operator оболочка прописана /bin/csh, и синтаксис там может отличаться.&lt;/p&gt;
&lt;p&gt;Не вопрос, топаем на удалённый сервер и делаем там:&lt;/p&gt;
&lt;p&gt;pw usermod backup_operator -s /bin/sh
Меняем, иными словами, ему оболочку на sh вместо csh. После этого конструкция&lt;/p&gt;
&lt;p&gt;ssh -i keyfile.ppk backup_operator@backup-ttr "ls /home/ | while read f; do echo ${f}; done;"
просто обязана была заработать. Но не заработала... А написала, что &lt;em&gt;f: Undefined variable.&lt;/em&gt;&lt;/p&gt;
&lt;p&gt;В процессе дальнейшего общения с коллективным разумом выяснилось следующее:&lt;/p&gt;
&lt;blockquote&gt;
&lt;p&gt;The shell will expand variables in double-quotes, but not in single-quotes.&lt;/p&gt;
&lt;/blockquote&gt;
&lt;p&gt;что если команду для удалённого исполнения заключать в двойные кавычки, то оболочка подставляет значение переменной вместо имени переменной. А в этой ситуации нужно, чтобы имя переменной передавалось как есть на дальнюю сторону, и значение вычислялось и использовалось уже там. Поэтому эту команду нужно заключать в ОДИНАРНЫЕ кавычки. То есть, вот так:&lt;/p&gt;
&lt;p&gt;ssh -i keyfile.ppk backup_operator@backup-ttr 'ls /home/ | while read f; do echo ${f}; done;'
Но есть и другой путь — просто ставить backslash перед символом переменной. То есть, можно сделать вот так:&lt;/p&gt;
&lt;p&gt;ssh -i keyfile.ppk backup_operator@backup-ttr "ls /home/ | while read f; do echo \${f}; done;'
Мне так пришлось сделать, потому что по итогу потребовалось, чтобы в одной команде были и локальные и удалённые переменные. Т. е. финальная рабочая конструкция выглядит вот так:&lt;/p&gt;
&lt;p&gt;ssh -i keyfile.ppk backup_operator@backup-ttr "cd ${backup_dir}; ls ./ | while read f; do mv \${f} \${f}.bak; done"&lt;/p&gt;</description><category>FreeBSD</category><category>Tips&amp;Tricks</category><category>Администрирование</category><category>Скрипты</category><guid>https://at-hacker.in/posts/udalyonnoe-vypolnenie-komand-po-ssh/</guid><pubDate>Wed, 20 Nov 2013 12:57:00 GMT</pubDate></item><item><title>Сеть по bluetooth с помощью btpand</title><link>https://at-hacker.in/posts/2010-01-07-2/</link><dc:creator>at-hacker</dc:creator><description>&lt;p&gt;В упомянутом уже ThinkPad 600X сетки не было. Точнее, сетевая PCMCIA была, но товарищ не нашёл проводочек, который подключается одним концом к сетевухе, а другим — к коннектору RJ-45. Поскольку CDROM в ноуте старый и довольно глючный, пришлось что-то придумывать. Вариант был ровно один — поскольку USB-порт там один есть, остаётся только Bluetooth, ну, или всякие USB-сетевые, проводные или WiFi. Покупать для этого ноута USB-сетевую меня заломало — не такая уж важная птица, а адаптеров голобуго зуба у меня навалом.  &lt;/p&gt;
&lt;p&gt;Под фрёй 7.2 я настроил сетку с помощью LAN over PPP, настроил rfcomm_pppd и полетели. Запускалось оно несколько криво — из rc.local, и требовало ещё не менее трёхсекундной паузы для прописывания маршрутизации, но в общем, работало.  &lt;/p&gt;
&lt;p&gt;После перехода на 8-ую фрю начали наблюдаться проблемы — примерно через час прослушивания интернет-радио голубозубый линк отваливался, и не поднимался до тех пор, пока я не выдерну и не вставлю обратно bluetooth-адаптер на серверной стороне, с последующим перезапуском rfcomm_pppd с обоих сторон. Разбираться с этим бесполезно — никаких сообщений в логах нет, ничего не происходит, сервер по голубому зубу при обзоре устройств виден, и линк поднимается, но трафика по нему не идёт. Я такое уже видел, правда, с КПК, но побороть так и не смог. Решил перейти на btpand, раз уж он появился — это софтина, портированная во фрю из NetBSD (не прошло и 8 лет, как фря наконец-то начала вкуривать BT PAN, ггг).   &lt;/p&gt;
&lt;p&gt;Потестил btpand при ручном запуске. На сервере запускаем:  &lt;/p&gt;
&lt;h2&gt;ifconfig tap0 create&lt;/h2&gt;
&lt;h2&gt;btpand -d ubt0 -i tap0 -s NAP&lt;/h2&gt;
&lt;h2&gt;ifconfig tap0 192.168.2.1 netmask 255.255.255.0&lt;/h2&gt;
&lt;p&gt;На клиенте:  &lt;/p&gt;
&lt;h2&gt;ifconfig tap0 create&lt;/h2&gt;
&lt;h2&gt;btpand -d ubt0 -i tap0 -s NAP -a 00:18:e7:1a:24:9b&lt;/h2&gt;
&lt;h2&gt;ifconfig tap0 192.168.2.2 netmask 255.255.255.0&lt;/h2&gt;
&lt;p&gt;Ну, тут всё понятно — создаём интерфейс tap0, через который будет работать голубозубый линк, поднимаем сам линк, и затем конфигурируем интерфейсы — прописываем IP-адрес, на котором оно будет работать.  &lt;/p&gt;
&lt;p&gt;Проверяем наличие присутствия связи:  &lt;/p&gt;
&lt;p&gt;notebook# ping -c 4 192.168.2.1&lt;br&gt;
PING 192.168.2.1 (192.168.2.1): 56 data bytes&lt;br&gt;
64 bytes from 192.168.2.1: icmp_seq=0 ttl=64 time=22.778 ms&lt;br&gt;
64 bytes from 192.168.2.1: icmp_seq=1 ttl=64 time=18.257 ms&lt;br&gt;
64 bytes from 192.168.2.1: icmp_seq=2 ttl=64 time=39.277 ms&lt;br&gt;
64 bytes from 192.168.2.1: icmp_seq=3 ttl=64 time=18.313 ms  &lt;/p&gt;
&lt;p&gt;Типа, работает, что ли? Афигеть... :-)  &lt;/p&gt;
&lt;p&gt;Ну чо, подадим на ноут интернет теперь — пропишем nameserver на ноуте в /etc/resolv.conf, а серверный голубозубый IP-шник в качестве шлюза и перенастроим правила файрвола, чтобы он ноут пускал в интернет:  &lt;/p&gt;
&lt;h2&gt;route add default 192.168.2.1&lt;/h2&gt;
&lt;p&gt;Проверяем:  &lt;/p&gt;
&lt;p&gt;notebook# ping ya.ru&lt;br&gt;
PING ya.ru (77.88.21.8): 56 data bytes&lt;br&gt;
64 bytes from 77.88.21.8: icmp_seq=0 ttl=57 time=85.294 ms&lt;br&gt;
64 bytes from 77.88.21.8: icmp_seq=1 ttl=57 time=74.188 ms&lt;br&gt;
64 bytes from 77.88.21.8: icmp_seq=2 ttl=57 time=217.258 ms  &lt;/p&gt;
&lt;p&gt;Круто! Грузим KDE и пытаемся открыть &lt;a href="http://shoutcast.com"&gt;http://shoutcast.com&lt;/a&gt; Получаем облом. Сайт грузится, но не до конца, и запустить прослушивание чего-нибудь оттуда невозможно — даже плейлист не качается. Интуиция подсказывает, что раз что-то работает, а что-то нет при живом линке, то это проблемы с MTU. Начинаю гонять пинги разного размера, определяя, какой максимальный размер пинга пройдёт. Оказалось, что пинги размером 641 байт проходят, и больше — хрен. А MTU на интерфейсах tap0 выставился дефолтный в 1500. Мораль — надо копать. Накопал только &lt;a href="http://lists.freebsd.org/pipermail/freebsd-bluetooth/2009-May/001417.html"&gt;вот это&lt;/a&gt;. Симптомы там у товарища в точности как у меня — MTU в районе 600 байт, иначе хеликоптер нихт. Но потом он перезагрузил телефон, к которому подключал комп по BT, и чудесным образом всё заработало с правильным MTU 1500. Но у меня такая маза не прокатила — сразу после перезагрузки оно вроде заработал с MTU 1500, но через пару минут всё отвалилось, в том числе ssh. Так что на текущий момент MTU 640 является обязательным.  &lt;/p&gt;
&lt;p&gt;Ну, теперь надо как-то организовать, чтобы вся эта хрень запускалась при старте системы. Условия у меня довольно специфические — в сервере BT-адаптер торчит всегда, а в ноуте — практически всегда, потому что я его в качестве плейера радива из интернета использую. Значит, надо либо в rc.local вписывать, что проще, но криво, либо написать rc-скрипт. Поскольку лёгких путей мы не ищем, то я выбрал второй вариант.  &lt;/p&gt;
&lt;p&gt;Скрипт будет принимать такие параметры:  &lt;/p&gt;
&lt;p&gt;&lt;strong&gt;btpand_enable&lt;/strong&gt; — разрешено или нет запускать btpand (по умолчанию NO);&lt;br&gt;
&lt;strong&gt;btpand_device&lt;/strong&gt; — девайс bluetooth, с которого работаем;&lt;br&gt;
&lt;strong&gt;btpand_interface — интерфейс, на котором работать (по умолчанию tap0);&lt;br&gt;
&lt;/strong&gt;btpand_service&lt;strong&gt;btpand_service&lt;/strong&gt; — сервис, который мы используем (GN, NAP, PANU, по умолчанию NAP);&lt;br&gt;
&lt;strong&gt;btpand_linkup&lt;/strong&gt; — скрипт, выполняемый при установлении соединения;&lt;br&gt;
&lt;strong&gt;btpand_linkdown&lt;/strong&gt; — скрипт, выполняемый при разрыве соединения;&lt;br&gt;
&lt;strong&gt;btpand_server&lt;/strong&gt; — MAC-адрес сервера, к которому будем подключаться, если работаем как клиент;&lt;br&gt;
&lt;strong&gt;btpand_iface_addr&lt;/strong&gt; — это параметр для конфигурирования интерфейса — прописывания IP-адреса и пр. Синтаксис такой же, как и у параметров rc.conf типа «ifconfig_rl0» — эта строка просто скармливается ifconfig'у.  &lt;/p&gt;
&lt;p&gt;Скрипты linkup/linkdown мне нужны, чтобы прописывать шлюз по умолчанию. В других задачах, возможно, потребуется ещё что-нибудь делать. Это необязательный параметр. Как, в общем, и многие другие. Обязательным будет являеться только btpand_enable и btpand_iface_addr для обоих компов, и btpand_server — только при запуске на клиенте. Остальное либо не обязательно, либо конфигурируется по умолчанию, как параметр btpand_interface.  &lt;/p&gt;
&lt;p&gt;Собственно, родился такой несложный скрипт:  &lt;/p&gt;
&lt;h2&gt;!/bin/sh&lt;/h2&gt;
&lt;h2&gt;PROVIDE: btpand&lt;/h2&gt;
&lt;h2&gt;REQUIRE: DAEMON sdpd&lt;/h2&gt;
&lt;h2&gt;BEFORE:  LOGIN&lt;/h2&gt;
&lt;h2&gt;KEYWORD: nojail&lt;/h2&gt;
&lt;p&gt;. /etc/rc.subr  &lt;/p&gt;
&lt;p&gt;name=«btpand»&lt;br&gt;
rcvar=&lt;code&gt;set_rcvar&lt;/code&gt;&lt;br&gt;
command=«/usr/sbin/btpand»  &lt;/p&gt;
&lt;p&gt;load_rc_config $name  &lt;/p&gt;
&lt;p&gt;eval «${rcvar}=\${${rcvar}:-'NO'}»  &lt;/p&gt;
&lt;p&gt;btpand_interface=${btpand_interface:-«tap0»}&lt;br&gt;
btpand_device=${btpand_device:-«ubt0hci»}&lt;br&gt;
btpand_service=${btpand_service:-«NAP»}&lt;br&gt;
btpand_server=${btpand_server:-«no»}&lt;br&gt;
btpand_iface_addr=${btpand_iface_addr:-«dhcp»}&lt;br&gt;
btpand_linkup=${btpand_linkup:-«no»}&lt;br&gt;
btpand_linkdown=${btpand_linkdown:-«no»}  &lt;/p&gt;
&lt;p&gt;start_cmd=«btpand_start»&lt;br&gt;
stop_precmd=«btpand_stopprecmd»&lt;br&gt;
stop_postcmd=«/sbin/ifconfig ${btpand_interface} destroy»  &lt;/p&gt;
&lt;p&gt;btpand_start()&lt;br&gt;
{&lt;br&gt;
  if [ ! -e /dev/${btpand_interface} ]&lt;br&gt;
    then /sbin/ifconfig ${btpand_interface} create&lt;br&gt;
  fi&lt;br&gt;
  /sbin/ifconfig ${btpand_interface} mtu 640&lt;br&gt;
  if [ «dhcp» = «${btpand_iface_addr}» ]&lt;br&gt;
    then /sbin/dhclient ${btpand_interface}&lt;br&gt;
    else /sbin/ifconfig ${btpand_interface} ${btpand_iface_addr}&lt;br&gt;
  fi&lt;br&gt;
  if [ «no» = «${btpand_server}» ]&lt;br&gt;
    then $command -d ${btpand_device} -s ${btpand_service} -i ${btpand_interface} &amp;gt; /dev/null 2&amp;gt;&amp;amp;1&lt;br&gt;
    else $command -d ${btpand_device} -s ${btpand_service} -i ${btpand_interface} \&lt;br&gt;
         -a ${btpand_server} &amp;gt; /dev/null 2&amp;gt;&amp;amp;1&lt;br&gt;
  fi&lt;br&gt;
  if [ ! «no» = «$btpand_linkup» ]&lt;br&gt;
    then sh ${btpand_linkup}&lt;br&gt;
  fi&lt;br&gt;
}  &lt;/p&gt;
&lt;p&gt;btpand_stopprecmd()&lt;br&gt;
{&lt;br&gt;
  if [ ! «no» = «$btpand_linkdown» ]&lt;br&gt;
    then sh ${btpand_linkdown}&lt;br&gt;
  fi&lt;br&gt;
}  &lt;/p&gt;
&lt;p&gt;run_rc_command «$1»&lt;/p&gt;
&lt;p&gt;Осталось это всё запустить. На сервере говорим:  &lt;/p&gt;
&lt;h2&gt;echo btpand_enable=«YES» &amp;gt;&amp;gt; /etc/rc.conf&lt;/h2&gt;
&lt;h2&gt;echo btpand_iface_addr=«192.168.2.1 netmask 255.255.255.0» &amp;gt;&amp;gt; /etc/rc.conf&lt;/h2&gt;
&lt;p&gt;На клиенте прописываем:  &lt;/p&gt;
&lt;h2&gt;echo btpand_enable=«YES» &amp;gt;&amp;gt; /etc/rc.conf&lt;/h2&gt;
&lt;h2&gt;echo btpand_iface_addr=«192.168.2.2 netmask 255.255.255.0» &amp;gt;&amp;gt; /etc/rc.conf&lt;/h2&gt;
&lt;h2&gt;echo btpand_server=«00:18:e7:1a:24:9b» &amp;gt;&amp;gt; /etc/rc.conf&lt;/h2&gt;
&lt;h2&gt;echo btpand_linkup=«/home/vasya/bt_linkup» &amp;gt;&amp;gt; /etc/rc.conf&lt;/h2&gt;
&lt;p&gt;И создаём файл /home/vasya/bt_linkup с таким содержанием:  &lt;/p&gt;
&lt;h2&gt;!/bin/sh&lt;/h2&gt;
&lt;p&gt;route add default 192.168.2.1  &lt;/p&gt;
&lt;p&gt;Перезапускаем сервер, перезапускаем ноут — судя по лампочкам на BT-адаптере, линк поднялся. Проверяем:  &lt;/p&gt;
&lt;p&gt;notebook# ping ya.ru&lt;br&gt;
PING ya.ru (77.88.21.8): 56 data bytes&lt;br&gt;
64 bytes from 77.88.21.8: icmp_seq=0 ttl=57 time=66.597 ms&lt;br&gt;
64 bytes from 77.88.21.8: icmp_seq=1 ttl=57 time=93.634 ms&lt;br&gt;
64 bytes from 77.88.21.8: icmp_seq=2 ttl=57 time=49.616 ms  &lt;/p&gt;
&lt;p&gt;Точно, поднялся. Ну, значит, топаем в KDE и запускаем чего-нибудь с Shoutcast'a бодрое на прослушивание! :-)  &lt;/p&gt;
&lt;p&gt;&lt;strong&gt;Область применения&lt;/strong&gt;  &lt;/p&gt;
&lt;p&gt;Вообще-то, это можно юзать не только для связи двух компов с фрёй между собой. BT-сервис «LAN over PPP» — довольно специфический, виндовые драйвера, которые я пробовал, его просто не видят, так что когда я захотел ноут с виндой, не имеющий WiFi адаптера, но имеющий BT, подключить в интернет через LAN over PPP, настроенный на серваке, то круто обломался.  &lt;/p&gt;
&lt;p&gt;Да и мобильные устройства далеко не все умеют LAN over PPP. Например, у меня был КПК HP hx2190 — он мог подключаться к такому сервису на сервере, и ходить через него в интернет и прослушивать музыку. А вот коммуникатор HTC Touch уже к серваку подключиться не мог — его BT-стек не поддерживает этот сервис. Touch, конечно, может работать по WiFi, который у меня всё равно в квартире есть, но WiFi гораздо более прожорлив в смысле батарейки, так что BT сильно предпочтительней, тем более что разницы в скорости между BT и WiFi линках на коммуникаторе практически незаметно.  &lt;/p&gt;
&lt;p&gt;К тому же, в коммуникаторе нет сервиса Dial-up internet, он интернет умеет раздавать только через PAN. При наличии btpand можно спокойно подключить 8-ую фрю к коммуникатору и заюзать интернет через EDGE.&lt;/p&gt;</description><category>Bluetooth</category><category>FreeBSD</category><category>Скрипты</category><guid>https://at-hacker.in/posts/2010-01-07-2/</guid><pubDate>Thu, 07 Jan 2010 02:30:00 GMT</pubDate></item></channel></rss>