<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>Юникс и другие</title>
	<atom:link href="http://boombick.org/blog/feed" rel="self" type="application/rss+xml" />
	<link>http://boombick.org/blog</link>
	<description>Just another WordPress weblog</description>
	<lastBuildDate>Wed, 26 May 2010 14:33:09 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.0</generator>
		<item>
		<title>Тестируем MariaDB</title>
		<link>http://boombick.org/blog/posts/99</link>
		<comments>http://boombick.org/blog/posts/99#comments</comments>
		<pubDate>Wed, 26 May 2010 14:33:09 +0000</pubDate>
		<dc:creator>boombick</dc:creator>
				<category><![CDATA[*NIX]]></category>
		<category><![CDATA[Debian]]></category>
		<category><![CDATA[PHP]]></category>

		<guid isPermaLink="false">http://boombick.org/blog/posts/99</guid>
		<description><![CDATA[На недавно прошедшей DevConf выступал с докладом Michael &#8220;Monty&#8221; Widenius, автор и идейный вдохновитель MySQL. Он рассказал про то, что Oracle потихоньку убивает MySQL и презентовал форк под названием MariaDB. Меня заинтересовала эта СУРБД и вот представляю вам свой опыт ее использования :) Итак, по шагам. Эксперементировать решил на Debian, коий и поднял в виртуальной [...]]]></description>
			<content:encoded><![CDATA[<p>На недавно прошедшей <a href="http://devconf.ru">DevConf</a> выступал с докладом <a href="http://en.wikipedia.org/wiki/Michael_Widenius">Michael &#8220;Monty&#8221; Widenius</a>, автор и идейный вдохновитель MySQL. Он рассказал про то, что Oracle потихоньку убивает MySQL и презентовал форк под названием <a href="http://askmonty.org/wiki/MariaDB_versus_MySQL">MariaDB</a>.<br />
Меня заинтересовала эта СУРБД и вот представляю вам свой опыт ее использования :)<br />
<span id="more-99"></span><br />
Итак, по шагам.<br />
Эксперементировать решил на Debian, коий и поднял в виртуальной машине. Меня, в первую очередь, интересовало ее применение на собственных проектах и совместимость с моими скриптами и сайтиками :)<br />
Итак, сначала выполняем следующие действия:</p>
<p>Добавляем ключ репозитория в хранилище и добавляем репозиторий в sources.list</p>
<blockquote><pre>
wget -O- http://ourdelta.org/deb/ourdelta.gpg | sudo apt-key add -
    sudo wget http://ourdelta.org/deb/sources/lenny-mariadb-ourdelta.list \
      -O /etc/apt/sources.list.d/ourdelta.list
</pre>
</blockquote>
<p>Затем устанавливаем, собственно, сервер и клиентскую либу. Все остальное, что нам будет нужно, потянется по зависимостям</p>
<blockquote><pre>
aptitude install libmariadbclient16 mariadb-server-5.1
 </pre>
</blockquote>
<p><a href='http://boombick.org/blog/wp-content/uploads/2010/05/password_setup.png' title='password_setup.png'><img src='http://boombick.org/blog/wp-content/uploads/2010/05/password_setup.png' alt='password_setup.png' /></a><br />
Все, как в mysql :) Вводим желаемый пароль root для входа в mariadb</p>
<p><a href='http://boombick.org/blog/wp-content/uploads/2010/05/login.png' title='login.png'><img src='http://boombick.org/blog/wp-content/uploads/2010/05/login.png' alt='login.png' /></a><br />
Затем логинимся</p>
<blockquote><pre>
mysql -u root -p
</pre>
</blockquote>
<p>Все до боли привычно и знакомо :)<br />
Ну чтож, пока все отлично (даже запускается как /etc/init.d/mysql start), проверим, как оно дружит с PHP</p>
<p>Сначала я хотел запустить стандартные бенчмарки, но жестоко обломался :) В репозитории были версии только для 64-битных систем<br />
<a href='http://boombick.org/blog/wp-content/uploads/2010/05/test_failed.png' title='test_failed.png'><img src='http://boombick.org/blog/wp-content/uploads/2010/05/test_failed.png' alt='test_failed.png' /></a></p>
<p>Ну и фиг с ним, подрубаем PHP</p>
<blockquote><pre>
aptitude install php5-cli php5-mysql
</pre>
</blockquote>
<p>создаем тестовую табличку<br />
<a href='http://boombick.org/blog/wp-content/uploads/2010/05/create_table.png' title='create_table.png'><img src='http://boombick.org/blog/wp-content/uploads/2010/05/create_table.png' alt='create_table.png' /></a><br />
и проверяем</p>
<p><a href='http://boombick.org/blog/wp-content/uploads/2010/05/test_connect.png' title='test_connect.png'><img src='http://boombick.org/blog/wp-content/uploads/2010/05/test_connect.png' alt='test_connect.png' /></a></p>
<p><a href='http://boombick.org/blog/wp-content/uploads/2010/05/test_connect_result.png' title='test_connect_result.png'><img src='http://boombick.org/blog/wp-content/uploads/2010/05/test_connect_result.png' alt='test_connect_result.png' /></a></p>
<p>Как видите, со стороны PHP все абсолютно прозрачно и никаких лишних телодвижений совершать не приходится. Это радует :)<br />
Для эксперимента можно поставить Wordpres<br />
Доставляем apache и все необходимое для него и пробуем<br />
<a href='http://boombick.org/blog/wp-content/uploads/2010/05/wp_install.png' title='wp_install.png'><img src='http://boombick.org/blog/wp-content/uploads/2010/05/wp_install.png' alt='wp_install.png' /></a></p>
<p><a href='http://boombick.org/blog/wp-content/uploads/2010/05/wp_installed.png' title='wp_installed.png'><img src='http://boombick.org/blog/wp-content/uploads/2010/05/wp_installed.png' alt='wp_installed.png' /></a></p>
<p>Все отлично :)</p>
<p>PS Моей целью не было выяснение всех отличий MariaDB от MySQL, меня интересовала именно возможность применять ее в собственных проектах и дальнейшего отказа от MySQL. Возможность проверил, в дальнейшем попробую поставить ее на реальный сервер и попробовать в  production на каких-нибудь не очень критичных проектах :)<br />
О результатх будет отдельный пост.</p>
]]></content:encoded>
			<wfw:commentRss>http://boombick.org/blog/posts/99/feed</wfw:commentRss>
		<slash:comments>3</slash:comments>
		</item>
		<item>
		<title>Консольные хитрости #2</title>
		<link>http://boombick.org/blog/posts/98</link>
		<comments>http://boombick.org/blog/posts/98#comments</comments>
		<pubDate>Mon, 24 May 2010 15:33:22 +0000</pubDate>
		<dc:creator>boombick</dc:creator>
				<category><![CDATA[*NIX]]></category>
		<category><![CDATA[shell]]></category>

		<guid isPermaLink="false">http://boombick.org/blog/posts/98</guid>
		<description><![CDATA[Продолжаем публикацию &#8220;однострочников&#8221; для bash, которые выполняют полезные и интересные действия :) Список установленных расширений для Firefox grep -hIr -m 1 em:name ~/.mozilla/firefox/*.default/extensions&#124;sed 's#\s*##'&#124;tr '=' '"""'&#124;cut -f3 -d'"'&#124;sort -u Список установленных расширений для Chromium (с адресом страницы) for i in $(find ~/.config/chromium/*/Extensions -name 'manifest.json'); do n=$(grep -hIr name $i&#124; cut -f4 -d '"'&#124; sort);u="https://chrome.google.com/extensions/detail/";ue=$(basename $(dirname [...]]]></description>
			<content:encoded><![CDATA[<p>Продолжаем публикацию &#8220;однострочников&#8221; для bash, которые выполняют полезные и интересные действия :)<br />
<span id="more-98"></span><br />
Список установленных расширений для Firefox</p>
<blockquote><pre>
grep -hIr -m 1 em:name ~/.mozilla/firefox/*.default/extensions|sed 's#\s*##'|tr '<>=' '"""'|cut -f3 -d'"'|sort -u
</pre>
</blockquote>
<p>Список установленных расширений для Chromium (с адресом страницы)</p>
<blockquote><pre>
for i in $(find ~/.config/chromium/*/Extensions -name 'manifest.json'); do n=$(grep -hIr name $i| cut -f4 -d '"'| sort);u="https://chrome.google.com/extensions/detail/";ue=$(basename $(dirname $(dirname $i))); echo -e "$n:\n$u$ue\n" ; done
</pre>
</blockquote>
<p>Уровень сигнала wifi</p>
<blockquote><pre>
/sbin/iwconfig wlan0 | grep Quality | cut -d = -f2 | awk '{print $1}'
</pre>
</blockquote>
<p>Оптимизация sqlite-баз, в которых Firefox хранит всякую служебную информацию. Позволяет немного ускорить его работу</p>
<blockquote><pre>
find ~/.mozilla/firefox -name '*.sqlite' -print -exec sqlite3 {} "VACUUM; REINDEX;" \;
</pre>
</blockquote>
<p>Количество процессов для пользователя (Linux only)</p>
<blockquote><pre>
pgrep -cu boombick
</pre>
</blockquote>
<p>Сравнить вывод двух команд</p>
<blockquote><pre>
diff <(tail -10 file1) <(tail -10 file2)
</pre>
</blockquote>
<p>Удалить все файлы, кроме одного :) Часто пригождается, чтобы не писать rm тут_куча-имен-файлов</p>
<blockquote><pre>
ls -Q | grep -v "not-for-delete-file" | xargs rm -r
</pre>
</blockquote>
<p>Собрать несколько тарболлов в один</p>
<blockquote><pre>
cat 1.tar.gz 2.tar.gz | tar zxvif -
</pre>
</blockquote>
<p>Размер окна терминала</p>
<blockquote><pre>
 echo $COLUMNS x $LINES
</pre>
</blockquote>
<p>Вывести man-страницу как обычный текст. Удобно для распечатки</p>
<blockquote><pre>
man need-man | col -bx
</pre>
</blockquote>
]]></content:encoded>
			<wfw:commentRss>http://boombick.org/blog/posts/98/feed</wfw:commentRss>
		<slash:comments>2</slash:comments>
		</item>
		<item>
		<title>Установка Django на devio.us</title>
		<link>http://boombick.org/blog/posts/95</link>
		<comments>http://boombick.org/blog/posts/95#comments</comments>
		<pubDate>Tue, 04 May 2010 08:02:31 +0000</pubDate>
		<dc:creator>boombick</dc:creator>
				<category><![CDATA[*NIX]]></category>
		<category><![CDATA[Programming]]></category>
		<category><![CDATA[Python]]></category>

		<guid isPermaLink="false">http://boombick.org/blog/posts/95</guid>
		<description><![CDATA[Как, наверняка, всем известно, недавно появился проект Devio.us, который предлагает всем желающим бесплатный shell-хостинг на базе OpenBSD. В нагрузку к шеллу также предлагается домен третьего уровня, вида %username%.devio.us или devio.us/~%username%, одна база данных MySQL и 100 мегов места. За пару баксов в месяц эти показатели можно улучшить :) На сервере установлен PHP (версии 5.2.12) как [...]]]></description>
			<content:encoded><![CDATA[<p>Как, наверняка, всем известно, недавно появился проект <a href="http://devio.us">Devio.us</a>, который предлагает всем желающим бесплатный shell-хостинг на базе OpenBSD. В нагрузку к шеллу также предлагается домен третьего уровня, вида %username%.devio.us или devio.us/~%username%, одна база данных MySQL и 100 мегов места. За пару баксов в месяц эти показатели можно улучшить :)<br />
На сервере установлен PHP (версии 5.2.12) как модуль апача (весьма, кстати, древнего &#8211; 1.3.29) и предлагается CGI-интерфейс. Ну и также доступны perl 5.10.1 и python 2.5.4. Целью проекта является привлечение новых членов в коммьюнити OpenBSD. При регистрации необходимо указать причину, по которой вам кровь из носу надо получить там аккаунт :) Процент отказов, к слову, довольно мал.<br />
Итак, поигравшись с PHP, захотелось чего-то большего. И так как на сервере присутствует python, захотелось прикрутить туда Django.<br />
<span id="more-95"></span><br />
<strong>Установка Django</strong><br />
Проведя ревизию модулей apache я выяснил, что ни mod_python, ни mod_wsgi не установлены. Грустно, но ладно. Для начала я просто залил туда дистрибутив django (версии 1.1.1)</p>
<blockquote>
<pre>
$ scp Django-1.1.1.tar.gz devio.us:</pre>
</blockquote>
<p>и распаковал его</p>
<blockquote>
<pre>
$ gunzip Django-1.1.1.tar.gz
$ tar xvf Django-1.1.1.tar
$ ln -s Django-1.1.1 django #  для дальнейшего удобства</pre>
</blockquote>
<p>Добавим переменную окружения PYTHONPATH для корректной работы Django</p>
<blockquote>
<pre>
$ export PYTHONPATH='/home/boombick/django'
$ echo "export PYTHONPATH='/home/boombick/django'" &gt;&gt; ~/.profile</pre>
</blockquote>
<p>И создадим симлинк для django-admin</p>
<blockquote>
<pre>
$ mkdir ~/bin
$ ln -s /home/boombick/django/django/bin/django-admin.py /home/boombik/bin/django-admin.py</pre>
</blockquote>
<p>Дальше все просто: создаем новый проект Django</p>
<blockquote>
<pre>
$ django-admin.py startproject djtest
$ cd djtest</pre>
</blockquote>
<p>И на этом этапе я просто запустил dev-сервер из поставки Django на высоком порту (занять порт с номером ниже 1024 без привелегий root не получится)</p>
<blockquote>
<pre>
$ python manage.py runserver 0.0.0.0:32000</pre>
</blockquote>
<p>Зайдя на http://boombick.devio.us:32000 я увидел приветственную страницу пустого проекта Django. Все получилось :)</p>
<p><strong>Запуск Django через CGI</strong><br />
В принципе, можно было бы оставить все и так. Но.<br />
1. Хотелось все-таки ходить к себе на страничку по 80 порту<br />
2. На free-аккаунте нельзя детачить приложения. Т.е. нельзя оставить запущенным сервер и разлогиниться. Что, впрочем, логично, иначе сервер превратился бы в рассадник IRC-ботов :)<br />
Можно, конечно, извратиться и держать логин постоянно, восстанавливая оборванное подключение и перезапуская сервер, но это не наш путь.</p>
<p>Итак.<br />
Сначала я просто пробрасывал запросы к Django через PHP (в виде index.php) :) Эдакий PHP-Proxy, но мало того, что это запредельно криво, так еще и не очень понятно, что делать с POST-запросами. Выход был один &#8211; заставить Django работать через CGI.<br />
Помучив гугл я нашел несколько разноненных мануалов и приступил к выполнению.<br />
Для начала создадим в директории <code>$HOME/public_html</code> файл django.cgi вот с таким содержанием:</p>
<blockquote>
<pre>
#!/usr/local/bin/python
# encoding: utf-8
"""
django.cgi

A simple cgi script which uses the django WSGI to serve requests.
Rewrited for use on devio.us shell hosting

Code copy/pasted from PEP-0333 and then tweaked to serve django.

http://www.python.org/dev/peps/pep-0333/#the-server-gateway-side

This script assumes django is on your sys.path, and that your site code is at
/home/mycode/mysite. Copy this script into your cgi-bin directory (or do
whatever you need to to make a cgi script executable on your system), and then
update the paths at the bottom of this file to suit your site.

This is probably the slowest way to serve django pages, as the python
interpreter, the django code-base and your site code has to be loaded every
time a request is served. FCGI and mod_python solve this problem, use them if
you can.

In order to speed things up it may be worth experimenting with running
uncompressed zips on the sys.path for django and the site code, as this can be
(theorectically) faster. See PEP-0273 (specifically Benchmarks).

http://www.python.org/dev/peps/pep-0273/

Make sure all python files are compiled in your code base. See

http://docs.python.org/lib/module-compileall.html

"""

import os, sys
sys.path.append("/home/boombick/django") # Поменяйте на путь к Django
sys.path.append("/home/boombick/djtest")  # Поменяйте на путь к вашему проекту

import django.core.handlers.wsgi

def run_with_cgi(application):

    environ                      = dict(os.environ.items())
    environ['wsgi.input']        = sys.stdin
    environ['wsgi.errors']       = sys.stderr
    environ['wsgi.version']      = (1,0)
    environ['wsgi.multithread']  = False
    environ['wsgi.multiprocess'] = True
    environ['wsgi.run_once']     = True

    if environ.get('HTTPS','off') in ('on','1'):
        environ['wsgi.url_scheme'] = 'https'
    else:
        environ['wsgi.url_scheme'] = 'http'

    headers_set  = []
    headers_sent = []

    def write(data):
        if not headers_set:
             raise AssertionError("write() before start_response()")

        elif not headers_sent:
             # Before the first output, send the stored headers
             status, response_headers = headers_sent[:] = headers_set
             sys.stdout.write('Status: %s\r\n' % status)
             for header in response_headers:
                 sys.stdout.write('%s: %s\r\n' % header)
             sys.stdout.write('\r\n')

        sys.stdout.write(data)
        sys.stdout.flush()

    def start_response(status,response_headers,exc_info=None):
        if exc_info:
            try:
                if headers_sent:
                    # Re-raise original exception if headers sent
                    raise exc_info[0], exc_info[1], exc_info[2]
            finally:
                exc_info = None     # avoid dangling circular ref
        elif headers_set:
            raise AssertionError("Headers already set!")

        headers_set[:] = [status,response_headers]
        return write

    result = application(environ, start_response)
    try:
        for data in result:
            if data:    # don't send headers until body appears
                write(data)
        if not headers_sent:
            write('')   # send headers now if body was empty
    finally:
        if hasattr(result,'close'):
            result.close()

os.environ['DJANGO_SETTINGS_MODULE'] = 'settings'
run_with_cgi(django.core.handlers.wsgi.WSGIHandler())</pre>
</blockquote>
<p>Затем меняем в <code>/path/to/dajngo-project/settings.py</code> параметр <code>ROOT_URLCONF</code> на <code>urls</code> вместо <code>projectname.urls</code><br />
Делаем файл исполняемым</p>
<blockquote>
<pre>
$ chmod a+x public_html/django.cgi</pre>
</blockquote>
<p>И проверяем работоспособность :)</p>
<p>http://boombick.devio.us/django.cgi</p>
<p><a href="http://boombick.org/blog/wp-content/uploads/2010/05/djangocgi.png" title="djangocgi.png"><img src="http://boombick.org/blog/wp-content/uploads/2010/05/djangocgi.thumbnail.png" alt="djangocgi.png" /></a><br />
Если все ок, то можно переходить к следующему шагу. А если нет и вы получаете 500 ошибку, то смотрите лог. Который, кстати, лежит в <code>/var/www/logs/error_log</code> В процессе установки очень удобно было его мониторить в отдельной консольке при помощи <code>tail -f</code><br />
Учтите только, что в этот лог падают сообщения о всех ошибках и вам надо отделять свои запросы от остальных :)</p>
<p><strong>Настройка Apache</strong><br />
Теперь надо настроить Apache для переадресации всех запросов к нашему django.cgi. Я очень давно не ковырялся с Apache (в последнее время использую исключительно nginx) и с трудом и доками вспоминал синтаксис mod_rewrite<br />
В результате родился вот такой вот файлик <code>.htaccess</code>, который надо положить в директорию <code>public_html</code></p>
<blockquote>
<pre>
RewriteEngine On
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule ^(.*)$ /django.cgi%{REQUEST_URI}?%{QUERY_STRING}</pre>
</blockquote>
<p><strong>Проверяем</strong><br />
В <code>~/djtest/urls.py</code> добавим правило для обработки всех адресов</p>
<blockquote>
<pre>
from django.conf.urls.defaults import *
from views import test_page

urlpatterns = patterns('',
    (r'^(.*)$', test_page)
)</pre>
</blockquote>
<p>И создадим файл <code>~/djtest/views.py</code> с, собственно, обработчиком</p>
<blockquote>
<pre>
from django.http import HttpResponse       

def test_page(request, url):
    html = "WORKING! Url is %s" % url
    return HttpResponse(html)</pre>
</blockquote>
<p>Заходим на http://boombick.devio.us/foo/bar<br />
<a href="http://boombick.org/blog/wp-content/uploads/2010/05/django.png" title="django.png"><img src="http://boombick.org/blog/wp-content/uploads/2010/05/django.thumbnail.png" alt="django.png" /></a><br />
Все работает :)</p>
<p><strong>Благодарности</strong><br />
Хочу сказать большое спасибо ребятам из конференции django@conference.jabber.ru, а конкретно <strong>ne_formal</strong> за неоценимую помощь :) Без него этот процесс затянулся бы куда больше!</p>
<p><strong>Важное замечание</strong><br />
Если вы читали комментарии в начале файла django.cgi, то не могли не заметить такой абзац:</p>
<blockquote><p>This is probably the slowest way to serve django pages, as the python<br />
interpreter, the django code-base and your site code has to be loaded every<br />
time a request is served. FCGI and mod_python solve this problem, use them if<br />
you can.
</p></blockquote>
<p>Что в вольном переводе выглядит как:<br />
<em>Это самый медленный путь для работы с django. На каждый запрос создается экземпляр интерпретатора Python. Если у вас есть возможность использовать FCGI или mod_python &#8211; используйте их!</em><br />
Так что статью прощу рассматривать исключительно как proof of concept или выход из совсем уж безвыходной ситуации! :)<br />
Enjoy!</p>
]]></content:encoded>
			<wfw:commentRss>http://boombick.org/blog/posts/95/feed</wfw:commentRss>
		<slash:comments>2</slash:comments>
		</item>
		<item>
		<title>Ловушки bash</title>
		<link>http://boombick.org/blog/posts/94</link>
		<comments>http://boombick.org/blog/posts/94#comments</comments>
		<pubDate>Thu, 29 Apr 2010 18:24:33 +0000</pubDate>
		<dc:creator>boombick</dc:creator>
				<category><![CDATA[Programming]]></category>
		<category><![CDATA[shell]]></category>

		<guid isPermaLink="false">http://boombick.org/blog/posts/94</guid>
		<description><![CDATA[Программирование на shell в общем и на bash в частности богато своими нюансами, которые, зачастую, упускаются из вида. В результате мы имеем проблемы на очевидных, вообщем-то, операциях. И как результат, зачастую, бывает &#8220;а ну его, этот баш! Перепишу на php/perl/python/ruby/etc&#8221; Эта статья написана для обсуждения и путей решения нескольких самых часто встречающихся &#8220;камней преткновения&#8221; при [...]]]></description>
			<content:encoded><![CDATA[<p>Программирование на shell в общем и на bash в частности богато своими нюансами, которые, зачастую, упускаются из вида. В результате мы имеем проблемы на очевидных, вообщем-то, операциях. И как результат, зачастую, бывает &#8220;а ну его, этот баш! Перепишу на php/perl/python/ruby/etc&#8221;<br />
Эта статья написана для обсуждения и путей решения нескольких самых часто встречающихся &#8220;камней преткновения&#8221; при программировании на bash. Я лично очень полюбил программировать на bash в последнее время и хочу поделиться кусочком знаний с вами :)<br />
<span id="more-94"></span><br />
<strong>1. for i in `ls *.mp3`</strong><br />
Знакомо? :) Если в имени файла встретится пробел, то все ваши усилия будут напрасны. Каждая из составляющих имени попадет в отдельную итерацию.</p>
<blockquote>
<pre>
 for i in `ls *.mp3`; do # Неверно!
    some command $i      # Неверно!
 done</pre>
</blockquote>
<p>Не получится и &#8220;закавычить&#8221; вывод <code>ls</code></p>
<blockquote>
<pre>
 for i in "`ls *.mp3`"; do # Неверно!
 ...</pre>
</blockquote>
<p>В этом случае ВЕСЬ вывод ls будет рассматриваться в контексте одной итерации. Это немного не то, чего хотелось бы добиться :) Решение есть</p>
<blockquote>
<pre>
for i in *.mp3; do  # Надо делать вот так и...
   some command "$i" # ...во втором пункте мы рассмотрим и это "узкое" место.
 done</pre>
</blockquote>
<p><strong>2. cp $file $target</strong><br />
Если в <code>$file</code> или <code>$target</code> окажутся пробелы, то вас ждет разочарование :)<br />
Выход не менее очевиден</p>
<blockquote>
<pre>
 cp "$file" "$target"</pre>
</blockquote>
<p><strong>3. Имена файлов, начинающиеся с дефисов</strong><br />
Всем известно, что параметры многих команд начинаются с дефиса <code>-</code>. В том случае, если с дефиса начинается имя файла, то оно будет ошибочно воспринято как параметр и вы получите ошибку. В лучшем случае.<br />
Одно из решений &#8211; поместить перед именами передаваемых фалов два дефиса <code>--</code>. Это сигнализирует команде (например <code>cp</code>) о том, что список параметров закончен и дальше идут аргументы:</p>
<blockquote>
<pre>
 cp -- "$file" "$target"</pre>
</blockquote>
<p>Но более элегантным решением, все-таки, будет цикл (причем с указанием каталога в пути к файлу):</p>
<blockquote>
<pre>
 for i in ./*.mp3; do
   cp "$i" /target
   ...</pre>
</blockquote>
<p>В этом случае аргумент, начинающийся с дефиса, будет передан как <code>./-foo.mp3</code> и все сработает нормально.</p>
<p><strong>4. [ $foo = "bar" ]</strong><br />
В bash вам необходимо заботиться о своих переменных. Иначе получите кучу ошибок :) Пример из заголовка выдаст ошибку в двух случаях:</p>
<ul>
<li>Если переменная, переданная в <code>[</code> не объявлена или пустая, то команда <code>[</code> "увидит" выражение<br />
<blockquote>
<pre>[ $foo = "bar" ]</pre>
</blockquote>
<p>как</p>
<blockquote>
<pre>[ = "bar" ]</pre>
</blockquote>
<p>и вы получите ошибку <em>unary operator expected</em> (ожидается унарный оператор). Так как оператор <code>=</code> бинарный, то <code>[</code> будет несколько шокирована :)</li>
<li>Если в <code>$foo</code> содержаться пробелы, то сравнение также будет некорректным<br />
<blockquote>
<pre>[ multiple words here = "bar" ]</pre>
</blockquote>
<p>И если вам это может показаться нормальным, то для <code>[</code> это довольно неожиданно :)</li>
</ul>
<p>Более корректно будет записать выражение как</p>
<blockquote>
<pre> [ "$foo" = bar ] # Все отлично</pre>
</blockquote>
<p>но, опять-таки, выполучите ошибку, если текст в переменной начинается с <code>-</code><br />
В bash есть ключевое слово <code>[[</code>, которое является расширением старой команды <code>test</code>, также известной как <code>[</code> и это решение всех подобных проблем :)</p>
<blockquote>
<pre>[[ $foo = bar ]] # Правильно</pre>
</blockquote>
<p>В случае с использованием <code>[[ ]]</code> вам не надо заключать переменную в кавычки, так как эта конструкция корректно обрабатывает и пустые переменные, и переменные, содержащие пробелы, и переменные, значение которых начинается с дефиса.<br />
Также вам может встретиться вот такой вариант:</p>
<blockquote>
<pre>[ x"$foo" = xbar ]</pre>
</blockquote>
<p><code>x"$foo"</code> - это хак для старых версий шелла, в которых вы вынуждены использовать <code>[</code>. И чтобы позаботиться о значении переменных, наичнающихся с дефиса, то можно использовать вот такую конструкцию.</p>
<p>А если одна из сторон сравниваемого выражения константа, то просто поместите переменную в правую часть :) <code>[</code> не обращает внимания на то, что находится справа</p>
<blockquote>
<pre>[ bar = "$foo" ]</pre>
</blockquote>
<p>Только левая часть выражения требует вашего особого внимания, на случай значения, начинающегося с дефиса.</p>
<p>Продолжение следует :)</p>
]]></content:encoded>
			<wfw:commentRss>http://boombick.org/blog/posts/94/feed</wfw:commentRss>
		<slash:comments>7</slash:comments>
		</item>
		<item>
		<title>PROMPT в стиле MS-DOS</title>
		<link>http://boombick.org/blog/posts/91</link>
		<comments>http://boombick.org/blog/posts/91#comments</comments>
		<pubDate>Mon, 29 Mar 2010 17:49:57 +0000</pubDate>
		<dc:creator>boombick</dc:creator>
				<category><![CDATA[*NIX]]></category>
		<category><![CDATA[shell]]></category>

		<guid isPermaLink="false">http://boombick.org/blog/posts/91</guid>
		<description><![CDATA[Наткнулся на забавную фишку :) Делаем следующее export PS1="C:\$( pwd &#124; sed 's:/:\\\\\:g' )> " и ностальгируем по временам MS-DOS :) Если очень понравилось, то можно добавить в ~/.bashrc следующий код: PS1="C:\$( pwd &#124; sed 's:/:\\\\\:g' )> " и наслаждаться красотой постоянно :)]]></description>
			<content:encoded><![CDATA[<p>Наткнулся на забавную фишку :) Делаем следующее</p>
<blockquote><pre>export PS1="C:\$( pwd | sed 's:/:\\\\\:g' )> "</pre>
</blockquote>
<p>и ностальгируем по временам MS-DOS :)</p>
<p><a href='http://boombick.org/blog/wp-content/uploads/2010/03/screen.png' title='PROMPT в стиле MS-DOS'><img src='http://boombick.org/blog/wp-content/uploads/2010/03/screen.thumbnail.png' alt='PROMPT в стиле MS-DOS' /></a></p>
<p>Если очень понравилось, то можно добавить в <code>~/.bashrc</code> следующий код:</p>
<blockquote><pre>
PS1="C:\$( pwd | sed 's:/:\\\\\:g' )> "
</pre>
</blockquote>
<p>и наслаждаться красотой постоянно :)</p>
]]></content:encoded>
			<wfw:commentRss>http://boombick.org/blog/posts/91/feed</wfw:commentRss>
		<slash:comments>7</slash:comments>
		</item>
		<item>
		<title>Консольные хитрости</title>
		<link>http://boombick.org/blog/posts/90</link>
		<comments>http://boombick.org/blog/posts/90#comments</comments>
		<pubDate>Sun, 28 Feb 2010 08:18:35 +0000</pubDate>
		<dc:creator>boombick</dc:creator>
				<category><![CDATA[*NIX]]></category>
		<category><![CDATA[shell]]></category>

		<guid isPermaLink="false">http://boombick.org/blog/posts/90</guid>
		<description><![CDATA[Опытные юниксоиды знают и осознают всю мощь использования командной строки и немного &#8220;свысока&#8221; посматривают на &#8220;мышевозов&#8221; :) Но и в консоли есть свои недостатки, которые побеждаются с помощью каких-то приемов. У каждого свой набор, а я хочу поделиться своими наработками-находками-привычками, которые появились за долгие годы использования командной строки. Эдакий небольшой cheat-sheet. Начнем :) 1. Копирование [...]]]></description>
			<content:encoded><![CDATA[<p>Опытные юниксоиды знают и осознают всю мощь использования командной строки и немного &#8220;свысока&#8221; посматривают на &#8220;мышевозов&#8221; :) Но и в консоли есть свои недостатки, которые побеждаются с помощью каких-то приемов. У каждого свой набор, а я хочу поделиться своими наработками-находками-привычками, которые появились за долгие годы использования командной строки. Эдакий небольшой cheat-sheet.<br />
Начнем :)</p>
<p>1. Копирование файлов с указанием прогресса. Всегда чертовски не хватало информативности в <code>cp</code></p>
<blockquote>
<pre>
rsync --progress file1 file2</pre>
</blockquote>
<p><span id="more-90"></span></p>
<p>2. Преобразование текста в картинку. Использовал для создания простенькой капчи</p>
<blockquote>
<pre>
echo -e "Some Text Line1\nSome Text Line 2" | convert -background none -density 196  -resample 72 -unsharp 0x.5 -font "Courier" text:- -trim +repage -bordercolor white -border 3  text.gif</pre>
</blockquote>
<p>3. Поиск и замена в текстовых файлах</p>
<blockquote>
<pre>
find . -iname "FILENAME" -exec sed -i 's/SEARCH_STRING/REPLACE_STRING/g' {} \;</pre>
</blockquote>
<p>4. Метод Столлмана! Отправка веб-странички по почте :)</p>
<blockquote>
<pre>
{ u="http://twitter.com/boombick"; echo "Subject: $u"; echo "Mime-Version: 1.0"; echo -e "Content-Type: text/html; charset=utf-8\n\n"; curl $u ; } | sendmail recipient@example.com</pre>
</blockquote>
<p>5. Запись iso-файла на диск</p>
<blockquote>
<pre>
cdrecord -v speed=4 driveropts=burnfree dev=/dev/scd0 cd.iso</pre>
</blockquote>
<p>6. Ограничение канала для scp. В данном случае ограничен 10kbps</p>
<blockquote>
<pre>
scp -l10 boombick@srv02:/home/svnstat/* .</pre>
</blockquote>
<p>7. Скриншот на удаленной машине :) (С запущенными иксами конечно)</p>
<blockquote>
<pre>
DISPLAY=":0.0" import -window root screenshot.png</pre>
</blockquote>
<p>8. Качаем и сразу распаковываем :)</p>
<blockquote>
<pre>
wget -qO - http://example.com/path/to/blah.tar.gz | tar xzf -</pre>
</blockquote>
<p>9. url_encode :) Достаточно примитивно, но часто помогает</p>
<blockquote>
<pre>
echo "$url" | sed 's/%/%25/g;s/ /%20/g;s/!/%21/g;s/"/%22/g;s/#/%23/g;s/\$/%24/g;s/\&amp;/%26/g;s/'\''/%27/g;s/(/%28/g;s/)/%29/g'</pre>
</blockquote>
<p>10. Погода в Москве :) Подставьте свой город</p>
<blockquote>
<pre>
curl -s "http://www.google.com/ig/api?weather=Moscow" | sed 's|.*&lt;temp_c data="\([^"]*\)"/&gt;.*|\1|'</pre>
</blockquote>
<p>Пока все, to be continued</p>
]]></content:encoded>
			<wfw:commentRss>http://boombick.org/blog/posts/90/feed</wfw:commentRss>
		<slash:comments>18</slash:comments>
		</item>
		<item>
		<title>Advanced grep</title>
		<link>http://boombick.org/blog/posts/88</link>
		<comments>http://boombick.org/blog/posts/88#comments</comments>
		<pubDate>Wed, 24 Feb 2010 12:48:17 +0000</pubDate>
		<dc:creator>boombick</dc:creator>
				<category><![CDATA[*NIX]]></category>
		<category><![CDATA[Programming]]></category>

		<guid isPermaLink="false">http://boombick.org/blog/posts/88</guid>
		<description><![CDATA[Написал небольшой скриптик, который красиво форматирует результаты рекурсивного поиска по исходникам (вообще-то по любым файлам, но я писал это все именно для поиска по дереву исходников :)) Вот так выглядят результаты его работы &#8211; это поиск слова &#8216;function&#8217; по всем файлам во всех вложенных подпапках Как видно на скрине, результаты поиска группируются по файлам, показывается [...]]]></description>
			<content:encoded><![CDATA[<p>Написал небольшой скриптик, который красиво форматирует результаты рекурсивного поиска по исходникам (вообще-то по любым файлам, но я писал это все именно для поиска по дереву исходников :))<br />
<span id="more-88"></span><br />
Вот так выглядят результаты его работы &#8211; это поиск слова &#8216;function&#8217; по всем файлам во всех вложенных подпапках<br />
<a href="http://boombick.org/blog/wp-content/uploads/2010/02/screen.png" title="screen.png"><img src="http://boombick.org/blog/wp-content/uploads/2010/02/screen.thumbnail.png" alt="screen.png" /></a><br />
Как видно на скрине, результаты поиска группируются по файлам, показывается на какой строке файла найдено совпадение, сама искомая строка подсвечивается.<br />
Поиск регистронезависимый.</p>
<p>@FIXME<br />
При подсветке слова регистр слова сбрасывается</p>
<p>@TODO<br />
Сделать разбивку результатов по страницам<br />
Вынести настройки в конфиг (например такие, как цвет в результатах поиска)</p>
<p>Для работы использует bash, awk, grep, sed.<br />
В скрипте есть такая строка</p>
<blockquote>
<pre>
grep -vE '(svn|jquery)'</pre>
</blockquote>
<p>Это, собственно, исключения, которые надо выбрасывать из результатов поиска. Т.е. строки, в которых эти &#8220;буквосочетания&#8221; есть, в результаты не попадут. Если вам это не нужно, просто выкиньте эту часть (в последствии это тоже уйдет в конфиг) :)</p>
<p>Изначально все писалось для FreeBSD, посему если у вас bash установлен не по пути <code>/usr/local/bin/bash</code>, то либо поменяйте путь к интерпретатору, либо сделайте симлинк :)</p>
<p>Скрипт распространяется под BSD-лицензией, о багах пишите сюда :)<br />
Скачать: <a href="http://code.google.com/p/srch/">http://code.google.com/p/srch/</a><br />
Вот, вообщем-то, и все, enjoy! Надеюсь, кому-нибудь пригодится :)<br />
UPD: Спасибо за совет с размещением на GoogleCode!</p>
]]></content:encoded>
			<wfw:commentRss>http://boombick.org/blog/posts/88/feed</wfw:commentRss>
		<slash:comments>12</slash:comments>
		</item>
		<item>
		<title>Хватит писать foreach()</title>
		<link>http://boombick.org/blog/posts/87</link>
		<comments>http://boombick.org/blog/posts/87#comments</comments>
		<pubDate>Fri, 19 Feb 2010 11:04:59 +0000</pubDate>
		<dc:creator>boombick</dc:creator>
				<category><![CDATA[Programming]]></category>

		<guid isPermaLink="false">http://boombick.org/blog/posts/87</guid>
		<description><![CDATA[По крайней мере в php :) Обращаюсь к php-разработчикам &#8211; подумайте, сколько раз за рабочий день вы пишите for() и foreach()? Я, например, довольно много :) Сейчас я пишу на php 5.3, а в нем есть очень удобные средства для того, чтобы обойтись без обхода массива в цикле &#8211; это функции для работы с массивами, [...]]]></description>
			<content:encoded><![CDATA[<p>По крайней мере в php :) Обращаюсь к php-разработчикам &#8211; подумайте, сколько раз за рабочий день вы пишите for() и foreach()? Я, например, довольно много :) Сейчас я пишу на php 5.3, а в нем есть очень удобные средства для того, чтобы обойтись без обхода массива в цикле &#8211; это функции для работы с массивами, которые принимают callback в качестве аргумента и <em>замыкания</em><br />
<span id="more-87"></span><br />
Я говорю о <a href="http://www.php.net/manual/en/function.array-map.php">array_map</a>(), <a href="http://www.php.net/manual/en/function.array-reduce.php">array_reduce</a>(), <a href="http://www.php.net/manual/en/function.array-filter.php%0A">array_filter</a>() и <a href="http://www.php.net/manual/en/function.uasort.php">uasort</a>() (ну и похожих на них). Использование этих функций &#8211; неплохой способ обойтись без обхода массивов в цикле.<br />
В PHP4 и PHP5.2 использование коллбэков выглядело весьма и весьма громоздко. Вы должны были определить функцию (или метод), а затем передать ее имя (или массив, содержащий имя класса и имя публичного или статического метода) в качестве аргумента. Можно было извернуться через create_function, но все равно это выглядело громоздко.<br />
В PHP5.3 вы можете определить анонимную функцию где-нибудь в коде (или даже непосредственно в вызове функции). Эти функции выглядят как обычные переменные и вы спокойно можете передавать их как аргумент.<br />
Я, конечно, не поклонник смешивания функционального и объектно-ориентированного подхода (тем более, что в PHP оба имеют проблемы с реализацией :)), но подобный прием поможет избавиться в коде от шума, который создают циклы, предназначенный для обхода массивов.</p>
<blockquote>
<pre>
// обычный массив простых чисел до 20
$primeNumbers = array(2, 3, 5, 7, 11, 13, 17, 19);

// array_map() применяет callback-функцию к каждому элементу массива и
// возвращает результат
$square = function($number) {
    return $number * $number;
};
$squared = array_map($square, $primeNumbers);
echo "Квадраты простых чисел: ",
     implode(', ', $squared), "\n";

// array_reduce() рекурсивно применяет фнукцию к парам значений
// и превращает массив в одно единственное значение
// Очень похоже на array_sum(),
// но использование callback открывает неплохие перспективы
$sum = function($a, $b) {
    return $a + $b;
};
$total = array_reduce($primeNumbers, $sum);
echo "Сумма простых чисел: ", $total, ".\n";

// array_filter() оставляет в массиве значения,
// которые удовлетворяют условию
$even = function($number) {
    return $number % 2 == 0;
};
$evenPrimes = array_filter($primeNumbers, $even);
echo "Четные простые числа: ",
     implode(', ', $evenPrimes), ".\n";

// uasort() сортирует массив и
// сохраняет связь междя ключом и значением
// похожа на asort(), но использует callback
$compare = function($a, $b) {
    if ($a == $b) {
        return 0;
    }
    return ($a &gt; $b) ? -1 : 1;
};
uasort($primeNumbers, $compare);
echo "Упорядоченный по убыванию массив простых чисел: ",
     implode(', ', $primeNumbers), ".\n";</pre>
</blockquote>
<p>UPDATE:<br />
Обещанные замеры производительности. Тест, конечно, весьма синтетический, но и он весьма нагляден. Я, честно говоря, был слегка удивлен.<br />
Код теста:</p>
<blockquote><pre>
echo "We're using the PHP " . phpversion() . "\n";

$array = $array2 = range(0, 10000, 1);

echo "Array contains " . count($array) . " elements\n";

$t1 = microtime(1);
echo "BEGIN array_map\n";

array_map(
        function($number){
                return $number * $number;
        },
        $array
);
$t2 = microtime(1);
$res = $t2 - $t1;
echo "END array_map: " . round($res, 4) . "\n\n";

$t1 = microtime(1);
echo "BEGIN foreach:\n";
foreach ($array2 as $key => $value){
        $array2[$key] = $value * $value;
}
$t2 = microtime(1);
$res = $t2 - $t1;
echo "END foreach: " . round($res, 4) . "\n\n";
</pre>
</blockquote>
<p>И вот результаты:</p>
<pre>
[boombick@srv01 /tmp]$ php -f test.php
We're using the PHP 5.3.1
Array contains 10001 elements
BEGIN array_map
END array_map: 0.0109

BEGIN foreach:
END foreach: 0.0052

[boombick@srv01 /tmp]$
</pre>
<p>Как видно из результата, array_map действительно проигрывает по скорости foreach</p>
]]></content:encoded>
			<wfw:commentRss>http://boombick.org/blog/posts/87/feed</wfw:commentRss>
		<slash:comments>19</slash:comments>
		</item>
		<item>
		<title>С Новым Годом!</title>
		<link>http://boombick.org/blog/posts/86</link>
		<comments>http://boombick.org/blog/posts/86#comments</comments>
		<pubDate>Thu, 31 Dec 2009 20:09:08 +0000</pubDate>
		<dc:creator>boombick</dc:creator>
				<category><![CDATA[boombick.org]]></category>

		<guid isPermaLink="false">http://boombick.org/blog/posts/86</guid>
		<description><![CDATA[Дорогие мои друзья и читатели! Поздравляю всех Вас с Новым годом, программистам желаю поменьше багов и легкой отладки, админам &#8211; хороших каналов, надежных провайдеров, умных юзеров и мощных серверов! Ну и все удачи, счастья и чтобы все у вас получалось! В этом году было довольно мало статей, за что извиняюсь перед вами и обещаю, что [...]]]></description>
			<content:encoded><![CDATA[<p>Дорогие мои друзья и читатели!<br />
Поздравляю всех Вас с Новым годом, программистам желаю поменьше багов и легкой отладки, админам &#8211; хороших каналов, надежных провайдеров, умных юзеров и мощных серверов! Ну и все удачи, счастья и чтобы все у вас получалось!<br />
В этом году было довольно мало статей, за что извиняюсь перед вами и обещаю, что в следующем году буду писать более-менее регулярно!<br />
С Новым Годом!</p>
]]></content:encoded>
			<wfw:commentRss>http://boombick.org/blog/posts/86/feed</wfw:commentRss>
		<slash:comments>2</slash:comments>
		</item>
		<item>
		<title>Немного о mod_perl</title>
		<link>http://boombick.org/blog/posts/83</link>
		<comments>http://boombick.org/blog/posts/83#comments</comments>
		<pubDate>Thu, 01 Oct 2009 13:18:24 +0000</pubDate>
		<dc:creator>boombick</dc:creator>
				<category><![CDATA[Debian]]></category>
		<category><![CDATA[web-servers]]></category>

		<guid isPermaLink="false">http://boombick.org/blog/posts/83</guid>
		<description><![CDATA[Apache &#8211; это очень популярный и широко используемый web-сервер. Есть множество альтернатив, но по возможностям настройки, гибкости и количеству доступных расширений Apache прочно держится в топе. В этой небольшой статье я расскажу как можно контролировать фактически все аспекты работы вашего web-сервера при помощи Perl и mod_perl mod_perl &#8211; это обычный модуль для Apache, который устанавливается [...]]]></description>
			<content:encoded><![CDATA[<p>Apache &#8211; это очень популярный и широко используемый web-сервер. Есть множество альтернатив, но по возможностям настройки, гибкости и количеству доступных расширений Apache прочно держится в топе. В этой небольшой статье я расскажу как можно контролировать фактически все аспекты работы вашего web-сервера при помощи Perl и <code>mod_perl</code><br />
<span id="more-83"></span><br />
<code>mod_perl</code> &#8211; это обычный модуль для Apache, который устанавливается также, как и все остальные модули. Но ключевое отличие в том, что этот модуль содержит в себе встроенный интерпретатор Perl, с помощью которого можно писать расширения и управлять фактически всеми аспектами работы сервера. Даже если вы никогда не писали (и не собираетесь писать) расширений для сервера, <code>mod_perl</code> пригодится вам чтобы, например, ускорить работу CGI-скриптов. А сам процесс описан ниже</p>
<p><b>Установка <code>mod_perl</code></b><br />
Все примеры показаны для ОС Linux Debian<br />
Устанавливаем пакет libapache2-mod-perl2</p>
<blockquote><pre>
lenny:~# apt-get install libapache2-mod-perl2
Reading package lists... Done
Building dependency tree
Reading state information... Done
The following extra packages will be installed:
  libapache2-reload-perl libbsd-resource-perl libperl5.10
The following NEW packages will be installed
  libapache2-mod-perl2 libapache2-reload-perl libbsd-resource-perl libperl5.10
0 upgraded, 4 newly installed, 0 to remove and 2 not upgraded.
Need to get 1201kB of archives.
After this operation, 3891kB of additional disk space will be used.
...
Processing triggers for man-db ...
Setting up libperl5.10 (5.10.0-19) ...
Setting up libapache2-mod-perl2 (2.0.4-5) ...
Enabling module perl.
Run '/etc/init.d/apache2 restart' to activate new configuration!
Setting up libapache2-reload-perl (0.10-2) ...
Setting up libbsd-resource-perl (1.28-1+b1) ...
</pre>
</blockquote>
<p>Как можно увидеть, модуль уже активирован.. Но никто не мешает сделать это еще раз (для надежности :) ) Или, например, после изменения конфигурации сервера, да и вообще, чтобы убедиться</p>
<blockquote><pre>lenny:~# a2enmod perl
Module perl already enabled
lenny:~# /etc/init.d/apache2 restart
Restarting web server: apache2 ... waiting .
</pre>
</blockquote>
<p>Модуль установлен, давайте займемся его настройкой. Самое простое &#8211; это добавить хендлер <code>perl-status</code> и посмотреть, что же происходит с нашим сервером. Создайте файл <code>/etc/apache2/conf.d/perl-status</code> со следующим содержимым:</p>
<blockquote><pre>
 #
 # make status information available at
 #   http://example.com/perl-status/
 #
 &lt;Location /perl-status&gt;
    SetHandler perl-script
    PerlHandler +Apache2::Status
&lt;/Location&gt;

 #
 # But only from the local host, and our trusted
 # remote IP.
 #
 &lt;Location /perl-status&gt;
    order deny,allow
    deny from all
    allow from 127.0.0.1
    allow from your.ip.address
&lt;/Location&gt;
</pre>
</blockquote>
<p>Перезапустите сервер и зайдите на адрес /perl-status. Узнаете несколько интересных вещей об окружении сервера и о загруженных модулях Perl<br />
<em><strong>Внимание!</strong></em> Эта информация содержит много такого, что не стоило бы показывать всем и каждому. Поэтому не стоит делать ее доступной для всех. Ограничьте доступ!</p>
<p><b>Ускорение Perl CGI-скриптов</b><br />
Так как <code>mod_perl</code> уже содержит встроенный интерпретатор Perl, то мы можем использовать его для запуска CGI-скриптов, вместо вызова внешней команды. Это позволит снизить накладные расходы на вызов отдельной копии интерпретатора при поступлении запроса на обработку скрипта. Есть способы добиться более внушительного ускорения, но это потребует внесения изменений в исходный код сценариев. В Debian CGI-скрипты по умолчанию располагаются в <code>/usr/lib/cgi-bin</code> Мы укажем Apache, что хотели бы запускать встроенный интерпретатор вместо внешнего для сценариев из этой директории.</p>
<blockquote><pre>
  PerlModule Apache::Registry
  Alias /cgi-bin/ /usr/lib/cgi-bin
  &lt;Location /cgi-bin&gt;
    SetHandler perl-script
    PerlHandler Apache::Registry
    PerlSendHeader On
    Options +ExecCGI
  &lt;/Location&gt;
</pre>
</blockquote>
<p>Перезапустите сервер (<code>/etc/init.d/apache2 restart</code>) и посмотрите, насколько быстрее стали работать скрипты :)</p>
<p><b>Загрузка модулей</b><br />
Продолжим. Давайте напишем код, который Apache мог бы использовать в процессе обработки страницы и пересылки ее клиенту. Нам необходимо разместить этот код так, чтобы встроенный интерпретатор смог загрузить его. По умолчанию в путь для поиска уже включены большинство системных директорий, но я считаю, что имеет смысл расположить его в очевидном месте :) <code>/etc/apache2/perl</code> Для добавления директории в путь поиска добавьте следующую директиву в <code>/etc/apache2/conf/00-mod-perl</code></p>
<blockquote><p>
#<br />
#  Load the startup script<br />
#<br />
PerlRequire /etc/apache2/perl/startup.pl</p>
</blockquote>
<p>Теперь при старте севера будет запускаться указанный скрипт. Давайте создадим его и поместим туда вот такой код</p>
<blockquote><p>
#<br />
#  Add /etc/apache2/perl to the mod_perl search path.<br />
#<br />
use lib  qw( /etc/apache2/perl );</p>
<p>#<br />
#  This script had no errors (?!)<br />
#<br />
1;
</p></blockquote>
<p>После перезапуска сервера путь для поиска будет обновлен и будет включать в себя нужную нам директорию.</p>
<p><b>Сделаем что-то полезное?</b><br />
Очень часто я вижу на своих проектах &#8220;пауков&#8221;, которые методично обшаривают страницу за страницей, качая содержимое. Вообщем я не возражаю против &#8220;зеркалирования&#8221; своих сайтов, но пока это все происходит незаметно для меня, в один поток и без съедания львиной доли ресурсов. А плохие &#8220;пауки&#8221; очень часто делают запросы, подобные следующим:</p>
<blockquote><p>
1.2.3.4 &#8211; [30/Apr/2009:19:36:29 +0100] &#8220;GET /articles/535#comment_2 HTTP/1.0&#8243;<br />
1.2.3.4 &#8211; [30/Apr/2009:19:36:30 +0100] &#8220;GET /articles/535#comment_3 HTTP/1.0&#8243;<br />
1.2.3.4 &#8211; [30/Apr/2009:19:36:30 +0100] &#8220;GET /articles/535#comment_5 HTTP/1.0&#8243;<br />
1.2.3.4 &#8211; [30/Apr/2009:19:36:31 +0100] &#8220;GET /articles/535#comment_4 HTTP/1.0&#8243;
</p></blockquote>
<p>Т.е. по якорям методично и многократно вытягивают одну и туже страницу, съедая ресурсы и трафик. Чтоже делать, будем их ловить с помощью вот такого модуля на Perl: <a href="http://boombick.org/DropClients.pm">DropClients</a><br />
Это сценарий на Perl, который обрабатывается <code>mod_perl</code>. Он выполняется для каждого входящего запроса и делает следующее:</p>
<ul>
<li>Выделяет из запроса IP-адрес и User-Agent</li>
<li>Если UA и IP &#8211; &#8220;плохие&#8221;, то отклоняет запрос</li>
<li>В нашем случае запрос отклоняется и тогда, когда в нем содержится &#8220;#&#8221;</li>
</ul>
<p>Если все хорошо, то запрос проходит</p>
<p>Чтобы запустить этот сценарий мы должны</p>
<ol>
<li>Загрузить его</li>
<li>И заставить его выполняться :)</li>
</ol>
<p>Сохраним модуль как <code>/etc/apache2/perl/DropClients.pm</code> и создадим файл <code>/etc/apache2/conf/dropclients</code> со следующим содержимым:</p>
<blockquote><p>
#<br />
#  Load the module<br />
#<br />
PerlModule DropClients</p>
<p>#<br />
#  Ensure it is invoked as an access handler<br />
#<br />
&lt;Location /&gt;<br />
     PerlAccessHandler DropClients<br />
&lt;/Location&gt;
</p></blockquote>
<p>После перезапуска метод <code>DropClients::request()</code> будет выполняться для каждого входящего запроса и нежелательные запросы будут отклоняться.</p>
<p>Нормальный запрос</p>
<blockquote><p>
user@notebook:~$ echo -e &#8220;GET / HTTP/1.0\n&#8221; | nc lenny 80 |grep ^HTTP<br />
HTTP/1.1 200 OK
</p></blockquote>
<p>&#8220;Плохой&#8221; запрос</p>
<blockquote><p>
user@notebook:~$ echo -e &#8220;GET /#foo HTTP/1.0\n&#8221; | nc lenny 80 |grep ^HTTP<br />
HTTP/1.1 403 Forbidden
</p></blockquote>
<p>Поскольку мы только что сделали &#8220;плохой&#8221; запрос, дальнейший доступ к серверу будет закрыт</p>
<blockquote><p>
user@notebook:~$ echo -e &#8220;GET / HTTP/1.0\n&#8221; | nc lenny 80 |grep ^HTTP<br />
HTTP/1.1 403 Forbidden
</p></blockquote>
<p>Если заглянете в код модуля, то увидите, что эта блокировка очень просто и снимается командой <code>rm -rf /tmp/blah</code></p>
<p><b>Различные типы обработчиков</b><br />
Для нашего модуля мы использовали обработчик <code>PerlAccessHandler</code>, который запускал скрипт при входящих соединениях. Но это лишь один из многих типов обработчиков. Есть также, например:</p>
<ul>
<li><code>PerlResponseHandler</code> &#8211; запускает сценарий после прохождения запроса</li>
<li><code>PerlOutputFilterHandler</code> &#8211; позволяет выполнить фильтрацию контента, отдаваемого пользователю (заменить слова, добавить футер&#8230; Сфера применения ограничена лишь вашей фантазией)</li>
</ul>
<p>Полный список обработчиков доступен в документации к <code>mod_perl</code> и <noindex><a href="http://perl.apache.org/docs/2.0/user/handlers/http.html">здесь</a></noindex> Там же можно найти и примеры кода</p>
]]></content:encoded>
			<wfw:commentRss>http://boombick.org/blog/posts/83/feed</wfw:commentRss>
		<slash:comments>1</slash:comments>
		</item>
	</channel>
</rss>
