А что это вообще такое?

Операторы ветвления

Операторы этого типа позволяют принимать решение о выполнении тех или иных действий в зависимости от ситуации.

В Parser существует два оператора ветвлений: if, проверяющий условие и выполняющий одну из указанных веток, и switch, выполняющий поиск необходимой ветки, соответствующей заданной строке или значению заданного выражения.

А что это вообще такое?

Чтение строки

$console:line

Такая конструкция считывает строку с консоли.




Чтение

$cookie:имя_cookie
Возвращает значение cookie с указанным именем.


Класс console[3.1.2]

Класс предназначен для создания простых интерактивных служб, работающих в текстовом построчном режиме.

Работать такие службы могут в паре со стандартной UNIX программой inetd.

Например, можно на Parser реализовать news-сервер (NNTP).
Добавьте такую строку в ваш /etc/inetd.conf и перезапустите inetd:
nntp stream tcp nowait учетная_запись /путь/к/parser3 /путь/к/parser3 /путь/к/nntp.p

В скрипте nntp.p опишите ваш NNTP сервер.
Что даст возможность людям его использовать - nntp://ваш_сервер.




Класс cookie

Класс предназначен для работы с HTTP cookies.




Кодировки

Уверены, наличие разных кодировок доставляет вам такое же удовольствие, как и нам.

В Parser встроена возможность прозрачного перекодирования документов из кодировки, используемой на сервере в кодировку посетителя и обратно.

Parser перекодирует
·данные форм;
·строки при преобразовании вида uri;
·текстовый результат обработки страницы.

Кодировку, используемую вами в документах на сервере, вы задаете в поле $request:charset.
Кодировку, желаемую вами в результате - в $response:charset.
Сделать это необходимо в одном из auto методов.

Рекомендуем задавать кодировку результата в HTTP заголовке content-type, чтобы программа просмотра страниц знала о ней, и пользователю вашего сервера не нужно было переключать ее вручную.
$response:content-type[
$.value[text/html]
$.charset[$response:charset]
]

Кодировку текста отправляемых вами писем можно задать отличной от кодировке результата, см. ^mail:send[…].

При работе с базами данных необходимо задать кодировку, в которой общаться с SQL-сервером, см. Формат строки подключения.

Список допустимых кодировок определяется в Конфигурационном файле.
По умолчанию везде используется кодировка UTF-8.

Примечание: имя кодировки нечувствительно к регистру. [3.1]




Создаст на две недели cookie

$cookie:login_name[
$.value[guest]
$.expires(14)
]

Создаст на две недели cookie с именем login_name и запишет в него значение guest.

записанные значения доступны для чтения

$cookie:my_cookie
Считывается и выдается значение cookie с именем my_cookie.

Примечание: записанные значения доступны для чтения сразу.

в него значение Петя. Созданный

$cookie:user[Петя]

Создаст cookie с именем user и запишет в него значение Петя. Созданный cookie будет храниться на диске пользователя 90 дней.

Примечание: записанное значение сразу доступно для чтения.

$cookie:имя[
$.value[значение]
$.expires(число дней)
]
$cookie:имя[
$.value[значение]
$.expires[session]
]
$cookie:имя[
$.value[значение]
$.domain[имя домена]
]
$cookie:имя[
$.value[значение]
$.path[подраздел]
]

Сохраняет в cookie с указанным именем.

Необязательные модификаторы:
$.expires(число дней) - задает число дней (может быть дробным, 1.5=полтора дня), на которое сохраняется cookie;
$.expires[session] - создает сеансовый cookie (cookie не будет сохранятся, а уничтожится с закрытием окна браузера);
$.expires[$date] - задает дату и время, до которой будет храниться cookie, здесь $date - переменная типа date;
$.domain[имя домена] - задает cookie в домен с указанным именем;
$.path[подраздел] - задает cookie только на определенный подраздел сайта.

Сборка Parser из исходных кодов

Загрузите из CVS исходные коды Parser3 и необходимых дополнительных модулей. Для этого выполните следующую команду:
cvs -d :pserver:anonymous@cvs.parser.ru:/parser3project login
Пароль пустой.

cvs -d :pserver:anonymous@cvs.parser.ru:/parser3project get -r имя_ветки имя_модуля

Имя_ветки - если не указывать -r, вы получите текущую разрабатываемую версию.
Для получения стабильной версии, забирайте ветку «release_3_X_XXXX».

Имя модуля:
Имя основного модуля: parser3.

Модуль с SQL драйверами: sql.
Сейчас в нем доступны каталоги:
sql/mysql
sql/pgsql
sql/oracle
sql/odbc

Для компиляции под UNIX…

…варианта Parser в виде модуля Apache 1.3 необходимо сначала в каталоге с исходными кодами Apache исполнить команду
./configure
и только после этого собирать Parser.

Для компиляции под Win32…

…необходим каталог:
win32/tools

…SQL драйверов необходимы каталоги:
win32/sql/mysql
win32/sql/pgsql
win32/sql/oracle

…варианта Parser, работающего с XML, в файле parser3/src/include/pa_config_fixed.h необходима директива
#define XML

…варианта Parser, принимающего письма по электронной почте, в файле parser3/src/include/pa_config_fixed.h необходима директива
#define WITH_MAILRECEIVE

Пользователи UNIX/Cygwin, инструкции по компиляции и установке читайте в файлах INSTALL каждого модуля.

Для компиляции под Win32 используйте Microsoft Visual Studio (6.0 или новее), используйте файлы .dsw каждого модуля. Распаковывайте все модули в каталог parser3project, находящийся в корне (важно!) диска.




Запись строки

$console:line[текст]

Такая конструкция выводит строку на консоль.




Запись

$cookie:имя[значение]
Сохраняет в cookie с указанным именем указанное значение на 90 дней.


А что это вообще такое?

Calendar. Создание календаря на заданную неделю месяца

^date:calendar[rus|eng](год;месяц;день)

Метод формирует таблицу с календарем на одну неделю заданного месяца года. Для определения недели используется параметр день. Параметр rus|eng также как и в предыдущем методе определяет формат календаря. С параметром rus дни недели начинаются с понедельника, c eng - с воскресенья.


Calendar. Создание календаря на заданный месяц

^date:calendar[rus|eng](год;месяц)

Метод формирует таблицу с календарем на заданный месяц года. Параметр rus|eng определяет формат календаря. С параметром rus дни недели начинаются с понедельника, c eng - с воскресенья.



Create. Дата или время в стандартном для СУБД формате

^date::create[год]
^date::create[год-месяц]
^date::create[год-месяц-день]
^date::create[год-месяц-деньчасов]
^date::create[год-месяц-день часов:минут]
^date::create[год-месяц-день часов:минут:секунд]
^date::create[год-месяц-день часов:минут:секунд.миллисекунд]
[3.1.3]
^date::create[часов:минут]
^date::create[часов:минут:секунд]

Создает объект класса date, содержащий значение произвольной даты и/или времени с точностью до секунды. Обязательными частями строки-параметра являются значение года или часа и минуты. месяц, день, часов, минут, секунд, миллисекунд являются необязательными, если не заданы, подставляются первый день, нулевые час, минута, секунда или текущий день.
Замечание: значение миллисекунд игнорируется.

Удобно использовать этот конструктор для работы с датами, полученными их базы данных, ведь из запроса вы получите значения полей с датой, временем или и датой и временем в виде строк.


Create. Относительная дата

^date::create(количествосуток после EPOCH)

Конструктор с одним параметром предназначен для задания относительных значений дат. Имея объект класса date, можно сформировать новый объект того же класса с датой, смещенной относительно исходной.


Create. Произвольная дата

^date::create(year;month)
^date::create(year;month;day)
^date::create(year;month;day;hour;minute;second)


Создает объект класса date, содержащий значение произвольной даты с точностью до секунды. Обязательными параметрами конструктора являются значения года и месяца. Параметры конструктора day,hour, minute, second являются необязательными, если не заданы, подставляются первый день, нулевые час, минута, секунда.


DOM. nodeType

DOM-элементы бывают разных типов, тип элемента хранится в integer поле nodeType.
В классе xdoc имеются следующие константы, удобные для проверки значения этого поля:

$xdoc:ELEMENT_NODE= 1
$xdoc:ATTRIBUTE_NODE = 2
$xdoc:TEXT_NODE = 3
$xdoc:CDATA_SECTION_NODE = 4
$xdoc:ENTITY_REFERENCE_NODE = 5
$xdoc:ENTITY_NODE = 6
$xdoc:PROCESSING_INSTRUCTION_NODE = 7
$xdoc:COMMENT_NODE = 8
$xdoc:DOCUMENT_NODE = 9
$xdoc:DOCUMENT_TYPE_NODE = 10
$xdoc:DOCUMENT_FRAGMENT_NODE = 11
$xdoc:NOTATION_NODE = 12



Класс date

Класс date предназначен для работы с датами. Возможные варианты использования - календари, всевозможные проверки, основывающиеся на датах и т.п.

Диапазон возможных значений - от 01.01.1970 до 01.01.2038 года.
Не забывайте, что в нашем календарном времени есть разрывы и нахлесты: во многих странах принято так-называемое «летнее» время, когда весной часы переводят вперед, а осенью назад.
Скажем, в Москве не бывает времени «02:00, 31 марта 2002», а время «02:00, 27 октября 2002» бывает дважды.

Числовое значение объекта класса date равно числу суток с EPOCH (01.01.1970 00:00:00, UTC) до даты, заданной в объекте. Этим значением полезно пользоваться для вычисления относительной даты, например:

# проверка "обновлен ли файл позже, чем неделю назад?"
^if($last_update > $now-7){
новый
}{
старый
}

Число суток может быть дробным, скажем, полтора дня = 1.5.

Обычно класс оперирует локальными датой и временем, однако можно узнать значение хранимой им даты/времени в произвольном часовом поясе, см. ^date.roll[TZ;…].

Для общения между компьютерами, работающими в разных часовых поясах, удобно обмениваться значениями даты/времени, не зависящими от пояса - здесь очень удобен UNIX формат, представляющий собой число секунд, прошедших с EPOCH.

Unix формат можно использовать в JavaScript и ряде других языков сценариев, работающих в браузере.

Parser полностью поддерживает работу с UNIX форматом дат.




Now. Текущая дата

^date::now[]
^date::now(смещениев днях)

Конструктор создает объект класса date, содержащий значение текущей даты с точностью до секунды, используя системное время сервера. Если указано, то плюс смещение в днях.

Используется локальное время той машины, где работает Parser (локальное время сервера). Для того, чтобы узнать время в другом часовом поясе, используйте ^date.roll[TZ;…].


Определение методов и пользовательских операторов

@имя[параметры]
тело


@имя[параметры][локальные переменные]

тело

Метод, это блок кода, имеющий имя, принимающий параметры, и возвращающий результат. Имена параметров метода перечисляются через точку с запятой. Метод также может иметь локальные переменные, которые необходимо объявить в заголовке метода, после объявления параметров, имена разделяются точкой с запятой.

Локальные переменные видны в пределах оператора или метода, и изнутри вызываемых ими операторов и методов, см. ниже $caller.

При описании метода можно пользоваться не только параметрами или локальными переменными, а также любыми другими именами, при этом вы будете работать с полями класса, или полями объекта, в зависимости от того, как был вызван определенный вами метод, статически, или динамически.

В Parser вы можете расширить базовый набор операторов, операторами в Parser считаются методы класса MAIN.


Определение пользовательского класса и операторов

Файл в таком формате определяет пользовательский класс:

@CLASS
имя_класса

#необязательно
@USE
файл_с_родительским классом

# необязательно
@BASE
имя_родительского_класса


# так рекомендуется называть метод-конструктор класса
@create[параметры]

# далее следуют определения методов класса
@method1[параметры]


Модуль можно подключить (см. «Подключение модулей») к произвольному файлу - там появится возможность использования определенного здесь класса.

Если не указать @CLASS, файл определит ряд дополнительных операторов.

Если определен метод…
@auto[]


…он будет выполнен автоматически при загрузке класса как статический метод (так называемый статический конструктор). Используется для инициализации статических полей (переменных) класса.
Примечание: результат работы метода игнорируется - никуда не попадает.

У метода auto может быть объявлен параметр:
@auto[filespec]

В этом параметре Parser передаст полное имя файла, содержащего метод.

В Parser создаваемые классы наследуют методы классов, от которых были унаследованы. Унаследованные методы можно переопределить.

В том случае, когда в качестве родительского класса выступает другой пользовательский класс, необходимо подключить модуль, в котором он находится, а также объявить класс базовым (@BASE).

Для того, чтобы пользоваться методами и полями родительских классов, необходимо использовать следующие конструкции:

^класс:метод[параметры] - вызов метода родительского класса (примечание: хотя такой синтаксис вызова метода и похож на синтаксис вызова статического метода, фактически, в случае динамического метода, происходит динамический вызов метода родительского класса), для обращения к своему ближайшему родительскому классу (базовому классу) можно использовать конструкции ^BASE::конструктор[параметры] и ^BASE:метод[параметры].


Поля объектов класса date

Через поля объектов класса date могут быть получены следующие величины:

$date.monthмесяц
$date.year год
$date.day день
$date.hour часы
$date.minute минуты
$date.second секунды
$date.weekday день недели (0 - воскресенье, 1 - понедельник, …)
$date.yearday день года (0 - 1-ое января, 1 - 2-ое января, …)
$date.daylightsaving 1 - летнее время, 0 - стандартное время
$date.TZ часовой пояс; [3.1.1]
содержит значение, если эта дата была получена ^date.roll[TZ;…]

и временем, которую сразу помещаем

$date_now[^date::now[]]
^connect[строка подключения]{
^void:sql{insert into access_log
(access_date)
values
('^date_now.sql-string[]')
}
}

Получаем строку вида '2001-11-30 13:09:56' с текущей датой и временем, которую сразу помещаем в колонку таблицы СУБД. Без использования данного метода пришлось бы выполнять необходимое форматирование вручную. Обратите внимание, данный метод не формирует кавычки, их требуется задавать вручную.

^if($node.nodeType

^if($node.nodeType == $xnode:ELEMENT_NODE){
<$node.tagName>
}

с календарем на ту неделю

$week_of_month[^date:calendar[rus](2001;11;30)]

В результате в переменную $week_of_month будет помещена таблица с календарем на ту неделю ноября 2001 года, которая содержит 30-е число. Формат таблицы следующий:

year

month

day

weekday

2001

11

26

01

2001

11

27

02

2001

11

28

03

2001

11

29

04

2001

11

30

05

2001

12

01

06

2001

12

02

07


$president_on_tv_at[^date::create(2001;12;31;23;55)]

В результате выполнения данного кода, создается объект класса date, значения полей которого соответствуют времени появления президента на телевизионном экране в комнате с веб-сервером.


$now[^date::now[]]
$date_after_week[^date::create($now+7)]

В примере получается дата, на неделю большая текущей.

Параметр конструктора не обязательно должен быть целым числом.
$date_after_three_hours[^date::create($now+3/24)]


# считаем новыми статьи за последние 3 дня
$new_after[^date::now(-3)]
$articles[^table::sql{select id, title, last_update from articles where …}]
^articles.menu{
$last_update[^date::create[$articles.last_update]]
$articles.title
^if($last_update > $new_after){новая}


}

Внимание пользователям Oracle: чтобы получать дату и время в удобном формате, в строке соединения с сервером укажите формат даты и времени, рекомендованный в Приложении 3.


$date_now[^date::now[]]
$date_now.year

$date_now.month

$date_now.day

$date_now.hour

$date_now.minute

$date_now.second

$date_now.weekday

В результате выполнения данного кода, создается объект класса date, содержащий значение текущей даты, а на экран будет выведено значение:
год
месяц
день
час
минута
секунда
день недели


$date_now[^date::now[]]
$date_now.month

В результате выполнения данного кода, создается объект класса date, содержащий значение текущей даты, а на экран будет выведен номер текущего месяца.


$date_now[^date::now[]]
^connect[строка подключения]{
^void:sql{insert into access_log
(access_date)
values
('^date_now.sql-string[]')
}
}

Получаем строку вида '2001-11-30 13:09:56' с текущей датой и временем, которую сразу помещаем в колонку таблицы СУБД. Без использования данного метода пришлось бы выполнять необходимое форматирование вручную. Обратите внимание, данный метод не формирует кавычки, их требуется задавать вручную.


^if($node.nodeType == $xnode:ELEMENT_NODE){
<$node.tagName>
}



В результате выполнения данного кода,

$president_on_tv_at[^date::create(2001;12;31;23;55)]

В результате выполнения данного кода, создается объект класса date, значения полей которого соответствуют времени появления президента на телевизионном экране в комнате с веб-сервером.

В примере получается дата, на

$now[^date::now[]]
$date_after_week[^date::create($now+7)]

В примере получается дата, на неделю большая текущей.

Параметр конструктора не обязательно должен быть целым числом.
$date_after_three_hours[^date::create($now+3/24)]

Внимание пользователям Oracle: чтобы получать

# считаем новыми статьи за последние 3 дня
$new_after[^date::now(-3)]
$articles[^table::sql{select id, title, last_update from articles where …}]
^articles.menu{
$last_update[^date::create[$articles.last_update]]
$articles.title
^if($last_update > $new_after){новая}


}

Внимание пользователям Oracle: чтобы получать дату и время в удобном формате, в строке соединения с сервером укажите формат даты и времени, рекомендованный в Приложении 3.

В результате выполнения данного кода,

$date_now[^date::now[]]
$date_now.year

$date_now.month

$date_now.day

$date_now.hour

$date_now.minute

$date_now.second

$date_now.weekday

В результате выполнения данного кода, создается объект класса date, содержащий значение текущей даты, а на экран будет выведено значение:
год
месяц
день
час
минута
секунда
день недели

В результате выполнения данного кода,

$date_now[^date::now[]]
$date_now.month

В результате выполнения данного кода, создается объект класса date, содержащий значение текущей даты, а на экран будет выведен номер текущего месяца.

Пример сдвига часового пояса

@main[]
$now[^date::now[]]
^show[]
^show[Москва;MSK-3MSD]
^show[Амстердам;MET-1DST]
^show[Лондон;GMT0BST]
^show[Нью-Йорк;EST5EDT]
^show[Чикаго;CST6CDT]
^show[Денвер;MST7MDT]
^show[Лос-Анжелес;PST8PDT]

@show[town;TZ]
^if(def $town){
$town
^now.roll[TZ;$TZ]
}{
Локальное время сервера
}


$now.year/$now.month/$now.day, $now.hour ч. $now.minute мин.





Пример сдвига месяца

$today[^date::now[]]
^today.roll[month](-1)
$today.month

В данном примере мы присваиваем переменной $today значение текущей даты и затем уменьшаем номер текущего месяца на единицу. В результате мы получаем номер предыдущего месяца.


Работа с переменными в динамических методах

Поиск значения переменной происходит в:
·в списке локальных переменных;
·в текущем объекте и его классе;
·в родительских объектах и их классах.

Запись значения переменной производится в уже имеющуюся переменную (см. область поиска выше), если таковая имеется. В противном случае создается новая переменная (поле) в текущем объекте.

Примечание: старайтесь всячески избегать использования полей класса не из методов класса, кроме простейших случаев! По-возможности, общайтесь с объектом только через его методы.




Работа с переменными в статических методах

Поиск значения переменной происходит в:
·в списке локальных переменных;
·в текущем классе или его родителях.

Запись значения переменной производится в уже имеющуюся переменную (см. область поиска выше), если таковая имеется. В противном случае создается новая переменная (поле) в текущем классе.


Roll. Сдвиг даты

^date.roll[year](смещение)
^date.roll[month](смещение)
^date.roll[day](смещение)

^date.roll[TZ][Новыйчасовой пояс] [3.1.1]

С помощью этого метода можно увеличивать/уменьшать значения полей year, month, day объектов класса date.

Также можно узнать дату/время, соответствующие хранящимся в объекте класса date в другом часовой поясе, задав системное имя нового часового пояса. Список имен см. в документации на вашу операционную систему, ключевые слова: «Переменная окружения TZ».


Системная переменная: caller

Все методы и операторы имеют локальную переменную caller, которая хранит «контекст вызова» метода или оператора.

Через нее можно:
·узнать, кто вызвал вызвавший описываемый метод или оператор, обратившись к $caller.self;
·считать - $caller.считать, или записать - $caller.записать[значение] переменную, как будто вы находитесь в том месте, откуда вызвали описываемый метод или оператор.

Например вам нужен оператор, похожий на системный for, но чем-то отличающийся от него. Вы можете написать его сами, воспользовавшись возможностью менять локальную переменную с именем, переданным вам, в контексте вызова вашего оператора.
@steppedfor[name;from;to;step;code]
$caller.$name($from)
^
while($caller.$name<=$to){
$code
^caller.$name.
inc($step)
}

Теперь такой вызов…
@somewhere[][i]
^steppedfor[i](1;10;2){$i }

…напечатает «1 3 5 7 9 », обратите внимание, что изменяется локальная переменная метода somewhere.

Примечание: возможность узнать контекст вызова удобна для задания контекста компиляции кода (см. «process. Компиляция и исполнение строки».




Системная переменная: result

Все методы и операторы имеют локальную переменную result. Если ей присвоить какое-то значение, то именно оно будет результатом выполнения метода. Значение переменной result можно считывать и использовать в вычислениях.

Пример:
@main[]
$a(2)
$b(3)
$summa[^sum[$a;$b]]
$summa

@sum[a;b]
^eval($a+$b)
$result[
Ничегоне скажу!]

Здесь клиенту будет выдана строка Ничего не скажу!, а не результат сложения двух чисел.

Важно: операторы, это методы класса MAIN, но в отличие от методов других классов, их можно вызвать из любого класса просто по имени, т.е. можно писать ^include[…], вместо громоздкого ^MAIN:include[…].


Системная переменная: self

Все методы и операторы имеют локальную переменную self, она хранит ссылку на текущий объект, в статических методах хранит то же, что и $CLASS.

Пример:
@main[]
$a[
Статическое поле ^$a класса MAIN]
^test[
Параметр метода]

@test[a]
^$a
- $a

^$self.a - $self.a

Выведет:
$a - Параметр метода
$self.a - Статическое поле $a класса MAIN



Системное поле класса: CLASS

$имя_класса:CLASS - хранит ссылку на класс объекта.

Это удобно при задании контекста компиляции кода (см. «process. Компиляция и исполнение строки».

По этой ссылке также доступны статические поля класса, пример:

@main[]
^method[$cookie:CLASS]

@method[storage]
$storage.field

Этот код напечатает значение $cookie:field.




Sql-string. Преобразование даты

^date.sql-string[]

Метод преобразует дату к виду ГГГГ-ММ-ДДЧЧ:ММ:СС, который принят для хранения дат в СУБД. Использование данного метода позволяет вносить в базы данных значения дат без дополнительных преобразований.


Unix-timestamp. Дата и время в UNIX формате[3.1.2]

^date::unix-timestamp(дата_время_в_UNIX_формате)

Конструктор создает объект класса date, содержащий значение, соответствующее переданному числовому значению в UNIX формате (см. также краткое описание).




Unix-timestamp. Преобразование

^date.unix-timestamp[]

Преобразует дату и время к значению в UNIX формате (см. краткое описание).




А что это вообще такое?

Класс env

Класс предназначен для получения значения переменных окружения. Со списком стандартных переменных окружения можно ознакомиться по адресу http://www.w3c.org/cgi. Веб-сервер Apache задает значения ряда дополнительных переменных.




Литералы в выражениях

Если строка содержит пробелы, то в выражении ее нужно заключать в кавычки или апострофы.

Пример:

^if($name eq Вася)
Здесь Вася - строка, которая не содержит пробелов, поэтому ее можно не заключать в кавычки или апострофы.

^if($name eq "Вася Пупкин")
Здесь строка содержит пробелы, поэтому заключена в кавычки.




Обработка ошибок

Человек несовершенен. Вы должны быть готовы к тому, что вместо ожидаемого, на экране вашего компьютера появится сообщение об ошибке. Избежать этого, к сожалению, почти невозможно. На начальном этапе сообщения об ошибках будут довольно частыми. Основной причиной ошибок сначала, вероятнее всего, будут непарные скобки (помните, мы говорили о текстовых редакторах, помогающих их контролировать?) и ошибки в записи конструкций Parser.

Если в ходе работы возникла ошибка, обычно обработка страницы прекращается, происходит откат (rollback) по всем активным на момент ошибки SQL-соединениям, и, вместо того вывода, который должен был попасть пользователю, вызывается метод unhandled_exception, ему передается информация об ошибке и стек вызовов, приведших к ошибке, и выдаются результаты его работы. Также ошибка записывается в журнал ошибок веб-сервера.

Однако часто желательно перехватить возникшую ошибку, и сделать нечто полезное вместо ошибочного кода.
Допустим, вы хотите проверить на правильность XML код, полученный из ненадежного источника. Здесь прерывание обработки страницы вам совершенно ни к чему, наоборот - вы ожидаете ошибку определенного типа и хотите ее обработать.
Parser с радостью идет навстречу, и дает вам в руки мощный инструмент: оператор try.

При сложной обработке данных часто выясняется, что имеет место ошибка в методе, вызванном из другого, а тот, в свою очередь, из третьего. Как в такой ситуации просто сообщить и обработать ошибку?
Используйте оператор throw, чтобы сообщить об ошибке, и обработайте ошибку на верхнем уровне - и вам не придется проверять ее на всех уровнях вложенности вызовов методов.

Также во многих ситуациях сам Parser или его системные классы сообщают о ошибках, см. «Системные ошибки».




Операторы в выражениях и их приоритеты

Операторы в таблице перечислены в порядке убывания приоритета:

Оператор

Значение
высший приоритет
()
Группировка частей выражения
!
Логическая операция NOT
~
Побитовая инверсия (NOT)
-
Одиночный минус
*
Умножение
/
Деление
Внимание,
\
Целочисленное деление
деление на ноль
%
Остаток от деления
дает ошибку number.zerodivision.
+
Сложение
-
Вычитание
<<
Побитовый сдвиг влево
>>
Побитовый сдвиг вправо
&
Побитовая операция AND
|
Побитовая операция OR
!|
Побитовая операция XOR
is
Проверка типа
def
Определен ли объект?
in
Находится ли текущий документ в каталоге?
-f
Существует ли файл?
–d
Существует ли каталог?
==
Равно
!=
Неравно
eq
Строки равны
ne
Строки не равны
<
Меньше
>
Больше
<=
Меньше или равно
>=
Больше или равно
lt
Строка меньше
gt
Строка больше
le
Строка меньше или равна
ge
Строка больше или равна
&&
Логическая операция AND
второй операнд не вычисляется, если первый - ложь
||
Логическая операция OR
второй операнд не вычисляется, если первый - истина
!||
Логическая операция XOR
низший приоритет





Получение значения поля запроса

$env:HTTP_ПОЛЕ_ЗАПРОСА

Такая конструкция возвращает значение поля запроса, передаваемое браузером веб-серверу (по HTTP протоколу).


Поля запроса имеют имена

^if(^env:HTTP_USER_AGENT.pos[MSIE]>=0){
Пользователь, вероятно, использует MicroSoft Internet Explorer

}

Поля запроса имеют имена в верхнем регистре и начинающиеся с HTTP_, и знаки '-' в них заменены на '_'.
Подробности в документации на ваш веб-сервер.


$env:REMOTE_ADDR

Возвратит IP-адрес машины, с которой был запрошен документ.



с которой был запрошен

$env:REMOTE_ADDR

Возвратит IP-адрес машины, с которой был запрошен документ.

Статические поля. Получение значения переменной окружения

$env:переменная_окружения

Возвращает значение указанной переменной окружения.


@Unhandled_exception. Вывод необработанных ошибок

Если ошибка так и не была обработана ни одним обработчиком (см. оператор try), Parser вызывает метод unhandled_exception, ему передается информация об ошибке и стек вызовов, приведших к ошибке, и выдаются результаты его работы. Также ошибка записывается в журнал ошибок веб-сервера.

Хороший тон, это оформить сообщение об ошибке в общем дизайне вашего сайта.
А также проверить, и не показывать технические подробности вашим посетителям.

Рекомендуем поместить этот метод в Конфигурационный файл сайта.



А что это вообще такое?

Basename. Имя файла без пути

^file:basename[filespec]

Из полного пути к файлу (filespec) получает имя файла с расширением имени, но без пути.


Cgi и exec. Исполнение программы

^file::cgi[имяфайла]
^file::cgi[имя файла;env_hash]
^file::cgi[имя файла;env_hash;аргументы]
^file::exec[имя файла]
^file::exec[имя файла;env_hash]
^file::exec[имя файла;env_hash;аргументы]


Конструктор cgi создает объект класса file, содержащий результат исполнения программы в соответствии со стандартом CGI.
Внимание: перез запуском программы Parser изменяет текущий каталог на каталог с программой.

Заголовки, которые выдаст CGI-скрипт, конструктор поместит в поля класса file в ВЕРХНЕМ регистре. Например, если некий скрипт script.pl, среди прочего, выдает в заголовке строку field:value, то после работы конструктора
$f[^file::cgi[script.pl]],
обратившись к $f.FIELD, получим значение value.

Конструктор exec аналогичен file::cgi, но не отделяет HTTP-заголовки от текста, возвращаемого скриптом.

Имя файла - имя файла с путем.

Объект, созданный этими конструкторами, имеет дополнительные поля:
status - информация о статусе завершении программы (обычно 0 - программа завершилась успешно, не 0 - с ошибкой)
stderr - результат считывания стандартного потока ошибок

Пример:
$cgi_file[^file::cgi[new.cgi]]
$cgi_file.text

Выведет на экран результаты работы скрипта new.cgi.

Необязательные параметры конструкторов ^file::cgi и ^file::exec:
env_hash - хеш, в котором могут задаваться
·дополнительные переменные окружения, которые впоследствии будут доступны внутри исполняемого скрипта,
·а также переменная stdin, содержащая текст, передаваемый исполняемому скрипту в стандартном потоке ввода.
Внимание: можно задавать только стандартные CGI переменные окружения и переменные, имена которых начинаются с CGI_ или HTTP_ (допустимы латинские буквы в ВЕРХНЕМ регистре, цифры, подчеркивание, минус).
Внимание: при обработке HTTP POST запроса, при помощи конструкции $.stdin[$request:body] вы можете передать в стандартный поток ввода скрипта полученные вами POST-данные. [3.0.8, раньше они передавались по-умолчанию]



Content-type. MIME-тип файла

$файл.content-type

Поле может содержать MIME-тип файла. При выполнении CGI-скрипта (см. file::cgi) MIME-тип может задаваться CGI-скриптом, полем заголовка ответа «content-type». При загрузке (см. file::load) или получении информации о файле (см. file::stat) MIME-тип определяется по таблице $MAIN:MIME-TYPES (см. «Конфигурационный метод»), если в таблице расширение имени файла найдено не будет, будет использован тип «application/octet-stream».


Delete. Удаление файла с диска

^file:delete[путь]

Удаляет указанный файл.
Путь - путь к файлу

Если после удаления в каталоге больше ничего не осталось, каталог тоже удаляется(если это возможно).


Dirname. Путь к файлу

^file:dirname[filespec]

Из пути к файлу или каталогу (filespec) получает только путь.


Fields. Все поля формы

$form:fields

Такая конструкция возвращает хеш со всеми полями формы или параметрами, переданными через URL. Имена ключей хеша те же, что и у полей формы, значениями ключей являются значения полей формы.


Find. Поиск файла на диске

^file:find[файл]
^file:find[файл]{код,если файл не найден}

Метод возвращает строку (объект класса string), содержащую имя файла с путем от корня веб пространства, если он существует по указанному пути, либо в каталогах более высокого уровня. В противном случае выполняется заданный код, если он указан.


Fullpath. Полное имя файла от корня веб-пространства

^file:fullpath[имя файла]

Из имени файла получает полное имя файла от корня веб-пространства. См. также «Приложение 1. Пути к файлам и каталогам».

Пример: в странице /document.html вы создаете ссылку на картинку, но настоящий адрес запрошенного документа может быть иным, скажем, при применении модуля mod_rewrite веб-сервера Apache, если поставить относительную ссылку на картинку, она не будет отображена браузером, поскольку браузер относительные пути разбирает относительно к текущему запрашиваемому документу, и ничего не знает про то, что на веб-сервере использован mod_rewrite.

Поэтому удобно заменить относительное имя на полное:
$image[^image::measure[^file:fullpath[image.gif]]]
^image.html[]

Такая конструкция…


…создаст код, содержащий абсолютный путь.




Imap. Получение координат нажатия в ISMAP

$form:imap

Если пользователь нажал на картинку с атрибутом ISMAP, такая конструкция возвращает хеш с полями x и y, в которых доступны координаты нажатия.



Justext. Расширение имени файла

^file:justext[filespec]

Из полного пути к файлу (filespec) получает расширение имени файла без точки.


Justname. Имя файла без расширения

^file:justname[filespec]

Из полного пути к файлу (filespec) получает имя файла без пути и расширения имени.


Класс file

Класс file предназначен для работы с файлами. Объекты класса могут быть созданы различными способами:

1.методом POST через поле формы
.

2.одним из конструкторов класса file.

При передачи файлов клиентам (например, методом mail:send или через поле response:body) необходимо задавать HTTP-заголовок content-type. В Parser для определения типа файла по расширению его имени существует таблица MIME-TYPES, определенная в Конфигурационном методе (см. главу Настройка). По ней, в зависимости от расширения файла, Parser автоматически определяет нужный тип данных для передачи в строке content-type. Если тип данных не удается определить по таблице, используется тип application/octet-stream.

Для проверки существования файлов и каталогов есть специальные операторы.




Класс form

Класс form предназначен для работы с полями форм. Класс имеет статические поля, доступные только для чтения.

Для проверки заполнения формы и редактирования имеющихся записей из базы данных удобно использовать такой подход:

^if($edit){
# запись из базы
$record[^
table::sql{… where id=…}]
}{
# новая запись, ошибка при заполнении, необходимо вывести
# поля формы
$record[$form:
fields]
}





List. Получение оглавления каталога

^file:list[путь]
^file:list[путь;фильтр]

Формирует таблицу (объект класса table) с одним столбцом name, содержащим файлы и каталоги по указанному пути, имена которых удовлетворяют шаблону, если он задан.

фильтр - регулярное выражение (см. метод match класса string) для обобщения типа запрашиваемых файлов. Без указания фильтра будут выведены все найдены по заданному пути файлы


Load. Загрузка файла с диска или HTTP-сервера

^file::load[формат;имя файла]
^file::load[формат;имя файла;опции загрузки]
^file::load[формат;имя файла;новоеимя файла]

^file::load[формат;имя файла;новое имя файла;опции загрузки]

Загружает файл с диска или HTTP-сервера.

Формат - формат представления загружаемого файла. Может быть text(текстовый) или binary(двоичный). Различие между этими типами в разных символах переноса строк. Для PC эти символы 0D 0A. При использовании формата text при загрузке 0D отбросится за ненадобностью, при записи методом save добавится.

имя файла - имя файла с путем или URL файла на HTTP-сервере.

Необходимо иметь ввиду, что если в конструкторе задан параметр новое имя файла, его значение будет присвоено полю name. Этим параметром удобно пользоваться при использовании метода mail:send для передачи файла под нужным именем.

опции загрузки
- см. «Работа с HTTP-серверами».

Если файл был загружен с HTTP-сервера, поля заголовков HTTP-ответа в верхнем регистре доступны как поля объекта класса file.
Также доступно поле tables, это хеш, ключами которого являются поля заголовки HTTP-ответа в верхнем регистре, а значениями таблицы с единственным столбцом value, содержащими все значения одноименных полей HTTP-ответа. [3.1.1]


Lock. Эксклюзивное выполнение кода

^file:lock[имяфайла-блокировки]{код}

Код не выполняется одновременно, для обеспечения эксклюзивности используется файл-блокировки.


Move. Перемещение или переименование файла

^file:move[старое имя файла;новое имя файла]

Метод переименовывает или перемещает файл и каталог (для платформы Win32 объекты нельзя перемещать через границу диска). Новый каталог создается с правами 775. Каталог старого файла удаляется, если после выполнения метода он остается пустым.

Внимание: необходимо крайне осторожно относиться к возможности записи в веб-пространстве, поскольку возможностью что-нибудь куда-нибудь записать нередко пользуются современные геростраты.




Name. Имя файла

$файл.name

Поле содержит имя файла. Объект класса file имеет поле name, если пользователь закачал файл через поле формы. Также в конструкторе file::load может быть указано альтернативное имя файла.


Поля HTTP-ответа

Если файл был загружен с HTTP-сервера, поля заголовков HTTP-ответа доступны доступны, как поля объекта класса file:

$файл.ПОЛЕ_HTTP_ОТВЕТА (ЗАГЛАВНЫМИ БУКВАМИ)

Например: $файл.SERVER.

Если один заголовок повторяется в ответе несколько раз, все его значения доступны в поле tables [3.0.8, пока нет в 3.1.0]:

$.tables[
$.HTTP-ЗАГОЛОВОК[
таблица значений, единственный столбец value]
]


Пример:

$f[^file::load[binary;http://www.parser.ru]]
^f.tables.foreach[key;value]{
$key=^value.menu{$value.value}[
|]

}




Получение значения поля формы

$form:поле_формы

Такая конструкция возвращает значение поля формы. Возвращаемый объект может принадлежать либо классу file, если поле формы имеет тип file, либо классу string. Дальнейшая работа с объектом возможна только методами, определенными для соответствующих классов.

Поле без имени считается имеющим имя nameless.
Координаты нажатия пользователем на картинку с атрибутом ISMAP доступны через $form:imap.


^file:basename

^file:basename[/a/some.tar.gz]

…выдаст…

some.tar.gz


^file:delete[story.txt]


#имя файла
^file:dirname[/a/some.tar.gz]
#имя каталога…
^file:dirname[/a/b/]

Оба вызова выдадут:
/a


^file:justext[/a/some.tar.gz]

…выдаст…

gz


^file:justname[/a/some.tar.gz]

…выдаст…

some.tar


$list[^file:list[/;\.zip^$]]
^list.menu{
$list.name

}

Выведет имена всех архивных файлов с расширением имени .zip, находящихся в корневом каталоге веб-сервера.


^file:lock[/counter.lock]{
$file[^file::load[text;/counter.txt]]
$string[^eval($file.text+1)]
^string.save[/counter.txt]
}
Количество посещений: $string


В отсутствие блокировки, два одновременных обращения к странице могли вызвать увеличение счетчика… на 1, а не на 2:
·пришел первый;
·пришел второй;
·считал первый, значение счетчика 0;
·считал второй, значение счетчика 0;
·увеличил первый, значение счетчика 1;
·увеличил второй, значение счетчика 1;
·записал первый, значение счетчика 1;
·записал второй поверх только что записанного первым, значение счетчика 1, а не 2.

Внимание: всегда думайте об одновременно приходящих запросах. При работе с базами данных обычно есть встроенные в SQL-сервер средства для их корректной обработки.


^file:move[/path/file1;/file1]

Переместит файл file1 в корень веб-пространства.


^archive.save[text;/arch/archive.txt]

Пример сохранит объект класса file в текстовом формате под именем archive.txt в каталог /arch/.


$name[image.gif]
$file[^file::load[$name]]
^connect[строка соединения]{
^void:sql{insert into images (name, bytes) values ('$name', '^file.sql-string[]')}
}


$f[^file::stat[some.zip]]
Размер в байтах: $f.size

Год создания: $f.cdate.year

$new_after[^date::now(-3)]
Статус: ^if($f.mdate >= $new_after){новый;старый}


^form:fields.foreach[field;value]{
$field - $value
}[
]

Пример выведет на экран все поля формы и соответствующие значения.
Предположим, что URI страницы www.mysite.ru/testing/index.html?name=dvoechnik&mark=2. Тогда пример выдаст следующее:

name - dvoechnik
mark - 2


Выберите, чем вы увлекаетесь в свободное время:

Театром

Кино

Книгами




$hobby[$form:tables.hobby]
^if($hobby){
Ваши хобби:

^hobby.menu{
$hobby.field
}[
]
}{
Ничего не выбрано
}

Пример выведет на экран выбранные варианты или напишет, что ничего не выбрано.



^file:delete

^file:delete[story.txt]

Оба вызова

#имя файла
^file:dirname[/a/some.tar.gz]
#имя каталога…
^file:dirname[/a/b/]

Оба вызова выдадут:
/a

^file:justext

^file:justext[/a/some.tar.gz]

…выдаст…

gz

^file:justname

^file:justname[/a/some.tar.gz]

…выдаст…

some.tar

Выведет имена всех архивных файлов

$list[^file:list[/;\.zip^$]]
^list.menu{
$list.name

}

Выведет имена всех архивных файлов с расширением имени .zip, находящихся в корневом каталоге веб-сервера.

В отсутствие блокировки, два одновременных

^file:lock[/counter.lock]{
$file[^file::load[text;/counter.txt]]
$string[^eval($file.text+1)]
^string.save[/counter.txt]
}
Количество посещений: $string


В отсутствие блокировки, два одновременных обращения к странице могли вызвать увеличение счетчика… на 1, а не на 2:
·пришел первый;
·пришел второй;
·считал первый, значение счетчика 0;
·считал второй, значение счетчика 0;
·увеличил первый, значение счетчика 1;
·увеличил второй, значение счетчика 1;
·записал первый, значение счетчика 1;
·записал второй поверх только что записанного первым, значение счетчика 1, а не 2.

Внимание: всегда думайте об одновременно приходящих запросах. При работе с базами данных обычно есть встроенные в SQL-сервер средства для их корректной обработки.

Переместит файл file1

^file:move[/path/file1;/file1]

Переместит файл file1 в корень веб-пространства.

Пример сохранит объект класса file

^archive.save[text;/arch/archive.txt]

Пример сохранит объект класса file в текстовом формате под именем archive.txt в каталог /arch/.

$name[image.gif]

$name[image.gif]
$file[^file::load[$name]]
^connect[строка соединения]{
^void:sql{insert into images (name, bytes) values ('$name', '^file.sql-string[]')}
}

$f[^file::stat[some.zip]]

$f[^file::stat[some.zip]]
Размер в байтах: $f.size

Год создания: $f.cdate.year

$new_after[^date::now(-3)]
Статус: ^if($f.mdate >= $new_after){новый;старый}

Пример выведет на экран все

^form:fields.foreach[field;value]{
$field - $value
}[
]

Пример выведет на экран все поля формы и соответствующие значения.
Предположим, что URI страницы www.mysite.ru/testing/index.html?name=dvoechnik&mark=2. Тогда пример выдаст следующее:

name - dvoechnik
mark - 2

Пример выведет на экран выбранные

Выберите, чем вы увлекаетесь в свободное время:

Театром

Кино

Книгами




$hobby[$form:tables.hobby]
^if($hobby){
Ваши хобби:

^hobby.menu{
$hobby.field
}[
]
}{
Ничего не выбрано
}

Пример выведет на экран выбранные варианты или напишет, что ничего не выбрано.

Пример без указания пути



Допустим, этот код расположен в документе /news/sport/index.html, здесь ищется файл header.gif в каталоге /news/sport/, разработанный специально для раздела спортивных новостей. Если он не найден, и не существует /news/sport/header.gif, то используется стандартный заголовочный рисунок новостного раздела.


безымянное поле



Внутри show.html строка 123 доступна как $form:nameless.

обычное поле






^if(def $form:photo){
^form:photo.
save[binary;/upload/photos/beauty.gif]
}
^if(def $form:user){
Пользователь: $form:user

}

Сохранит картинку, выбранную пользователем в поле формы и присланную на сервер, в заданном файле.

Пример передачи нескольких аргументов

Кроме того, вызываемой программе можно передать ряд аргументов, перечислив их через точку с запятой после хеша переменных окружения:
^file::exec[script.pl;;длина;ширина]

Внимание: настоятельно рекомендуется хранить запускаемые скрипты вне веб-пространства, поскольку запуск скрипта с произвольными параметрами может привести к неожиданным результатам.




Пример с указанием пути



Здесь ищется файл header.gif в каталоге /i/раздел/подраздел/. Если он не найден, он будет последовательно искаться в каталогах
·/i/раздел/
·/i/
·/





Пример внешнего скрипта

^file::exec[script.pl;$.CGI_INFORMATION[этогомне не хватало]]

Внутри скрипта script.pl можно воспользоваться переданной информацией:
print "Дополнительная информация: $ENV{CGI_INFORMATION}\n";




Пример загрузки файла с диска

$f[^file::load[binary;article.txt]]
Файл с именем $f.name имеет размер $f.size и содержит текст:

$f.text

Выведет размер, имя и текст файла.


Пример загрузки файла с HTTP-сервера

$file[^file::load[text;http://parser.ru/]]
Программное обеспечение сервера: $file.SERVER


$file.text





Qtail. Получение остатка строки запроса

$form:qtail

Возвращает часть $request:query после второго ?.



Save. Сохранение файла на диске

^файл.save[формат;имя файла]

Метод сохраняет объект в файл в заданном формате под указанным именем.

Формат - формат сохранения файла (text или binary)
Имя файла - имя файла и путь, по которому он будет сохранен


Size. Размер файла

$файл.size

Поле содержит размер файла в байтах.


Sql-string. Сохранение файла на SQL-сервере[3.1.2]

^file.sql-string[]

Выдает строку, которую можно использовать в SQL-запросе. Позволяет сохранить файл в базе данных.

Внимание: на данный момент реализована поддержка только MySQL-сервера.


Sql. Загрузка файла из SQL-сервера[3.1.2]

^file::sql{запрос}
^file::sql[имя файла]{запрос}

Загружает файл из SQL-сервера. Результатом выполнения запроса должна быть одна запись. Считается, что ее
·первая колонка содержит данные файла;
·вторая колонка содержит имя файла;
·третья колонка содержит content-type файла (если не указан, он будет определен по таблице $MIME-TYPES).

Имя файла может быть также задано параметром.

Имя файла и его content-type будет переданы посетителю при $response:download.




Stat. Получение информации о файле

^file::stat[имяфайла]
Объект, созданный этим конструктором, имеет дополнительные поля (объекты класса date):

$файл.size - размер файла в байтах;
$файл.cdate - дата создания;
$файл.mdate - дата изменения;
$файл.adate - дата последнего обращения к файлу.

имя файла - имя файла с путем.


Tables. Получение множества значений поля

$form:tables

Такая конструкция возвращает хеш со всеми полями формы или параметрами, переданными через URL. Имена ключей хеша те же, что и у полей формы, значениями же являются таблицы, см. ниже.

$form:tables.поле_формы

Если поле формы имеет хотя бы одно значение, такая конструкция возвращает таблицу (объект класса table) с одним столбцом field, содержащим все значения поля. Используется для получения множества значений поля.
Внимание: не забудьте проверить наличие таблицы перед тем, как начать ею оперировать.


Text. Текст файла

$файл.text

Поле содержит текст файла. Использование этого поля позволяет выводить на странице содержимое текстовых файлов или результатов работы file::cgi и file::exec.


А что это вообще такое?

Add. Сложение хешей

^хеш.add[хеш-слагаемое]

Добавляет к хешу другой хеш-слагаемое, при этом одноименные ключи хеша перезаписываются.


Чтение

$hashfile.ключ

Возвращает строку, ассоциированную с ключом, если эта ассоциация не устарела.




Clear. Удаление всего содержимого

^hashfile.clear[]

Удаляет все ассоциации.




_Count. Количество ключей хеша

^хеш._count[]

Возвращает количество ключей хеша.


Create. Создание пустого и копирование хеша

^hash::create[]
^hash::create[существующий хеш или хешфайл]

Если параметр не задан, будет создан пустой хеш.
Если указан существующий хеш или хешфайл, конструктор создает его копию.

Пустой хеш, создаваемый конструктором без параметров, нужен в ситуации, когда необходимо динамически наполнить хеш данными, например:
$dyn[^hash::create[]]
^for[i](1;10){
$dyn.$i[$value]
}
Перед выполнением for мы определили, что именно наполняем.

Если предполагается интенсивная работа по изменению содержимого хеша, но необходимо сохранить, скажем, значения по умолчанию, например:
$pets[
$.pet[
Собака]
$.food[
Косточка]
$.good[
Ошейник]
]
$pets_copy[^hash::create[$pets]]





Delete. Удаление файлов данных с диска

^hashfile.delete[]

Удаляет с диска файлы, в которых хранятся данные хеш файла.




Delete. Удаление пары ключ/значение

^hashfile.delete[ключ]

Метод удаляет из файла пару ключ/значение.




Delete. Удаление пары ключ/значение

^хеш.delete[ключ]

Метод удаляет из хеша пару ключ/значение.


Foreach. Перебор ключей хеша

^хеш.foreach[ключ;значение]{тело}
^хеш.foreach[ключ;значение]{тело}[разделитель]
^хеш.foreach[ключ;значение]{тело}{разделитель}


Метод аналогичен методу menu класса table. Перебирает все ключи хеша и соответствующие им значения (порядок перебора не определен).

ключ - имя переменной, которая возвращает имена ключей
значение - имя переменной, которая возвращает соответствующие значения ключей
тело - код, исполняемый для каждой пары ключ-значение хеша
разделитель - код, который выполняется перед каждым непустым не первым телом


Foreach. Перебор ключей хеша

^hashfile.foreach[ключ;значение]{тело}
^hashfile.foreach[ключ;значение]{тело}[разделитель]
^hashfile.foreach[ключ;значение]{тело}{разделитель}


Перебирает все ключи хеша и соответствующие им значения (порядок перебора не определен). Метод аналогичен foreach класса hash.




Hash. Преобразование к обычному hash

^hashfile.hash[]

Преобразует hashfile в обычный хеш.




Хеш (ассоциативный массив)

Хеш, или ассоциативный массив - позволяет хранить ассоциации между строковыми ключами и произвольными значениями. Создание хеша происходит автоматически при таком присваивании переменной значения или вызове метода:

$имя[
$.ключ1[значение]
$.ключ2[значение]
. . .
$.ключN[значение]
]

или

^метод[
$.ключ1[значение]
$.ключ2[значение]
. . .
$.ключN[значение]
]

Также можно создать пустой копию другого хеша, см. «Класс hash, create. Создание пустого и копирование хеша».

Получение значений ключей хеша:
$имя.ключ

Хеш позволяет создавать многомерные структуры, например, hash of hash, где значениями ключей хеша выступают другие хеши.

$имя[
$.ключ1_уровня1[$.ключ1_уровня2[значение]]
. . .
$.ключN_уровня1[$.ключN_уровня2[значение]]
]





Intersection. Пересечение хешей

^хеш_a.intersection[хеш_b]

Метод выполняет пересечение двух хешей. Возвращает хеш, содержащий ключи, принадлежащие как хешу a, так и b. Результат необходимо присваивать новому хешу.


Intersects. Определение наличия пересечения хешей

^хеш_a.intersects[хеш_b]

Метод определяет наличие пересечения (одинаковых ключей) двух хешей. Возвращает булевое значение "истина", если пересечение есть, или "ложь" в противном случае.


Использование хеша вместо таблицы

$хеш.fields - сам хеш.

Для большей взаимозаменяемости таблиц и хешей поле fields хранит ссылку на сам хеш. См. table.fields.




Как работать с документацией

Данное руководство состоит из трех частей.

В первой, учебной части рассматриваются практические задачи, решаемые с помощью Parser. На примере создания учебного сайта показаны базовые возможности языка и основные конструкции. Для создания кода можно пользоваться любым текстовым редактором. Желательно, чтобы в нем был предусмотрен контроль над парностью скобок с параллельной подсветкой, поскольку при увеличении объема кода и его усложнении становится сложно следить за тем, к чему относится та или иная скобка, и эта возможность существенно облегчит ваш труд. Также поможет в работе и выделение цветом конструкций языка. Читать и редактировать код станет намного проще.

Учебная часть построена в виде уроков, в начале которых предлагается рабочий код, который можно просто скопировать в нужные файлы. Далее подробно разбирается весь пример с объяснением логики его работы. В конце каждого урока тезисно перечислены все основные моменты, а также даны рекомендации, на что надо обратить особое внимание. Внимательное изучение представленных уроков обеспечит вас необходимым запасом знаний для последующей самостоятельной работы над проектами на Parser.

Во второй части представлен справочник по синтаксису языка и подробно рассмотрены правила описания различных конструкций.

Третья часть представляет собой справочник операторов и базовых классов языка с описанием методов и краткими примерами их использования.

В приложениях к документации рассмотрены вопросы установки и конфигурирования Parser.




_Keys. Список ключей хеша

^хеш._keys[]
^хеш._keys[имя столбца] [3.1.2]

Метод возвращает таблицу (объект класса table), содержащую единственный столбец, где перечислены все ключи хеша (порядок перечисления не определен).
Имя столбца - «key» или переданное имя.


Класс hash

Класс предназначен для работы с хешами - ассоциативными массивами. Хеш считается определенным (def), если он не пустой. Числовым значением хеша является число ключей (значение, возвращаемое методом ^хеш._count[]).




Класс hashfile [3.1.2]

Класс предназначен для работы с хешами, хранящимися на диске. В отличие от класса hash объекты данного класса считаются всегда определенным (def) и не имеют числового значения.

Если класс hash хранит свои данные в оперативной памяти, hashfile хранит их на диске, причем можно отдельно задавать время хранения каждой пары ключ-значение.
Замечание: в настоящий момент для хранения одного hashfile используются два файла: .dir и .pag.

Чтение и запись данных происходит очень быстро - идет работа только с необходимыми фрагментами файлов данных.
На простых задачах hashfile работает значительно быстрее баз данных.
Замечание: в один момент времени с файл может изменяться только одним скриптом, остальные ждут окончания его работы.




Конструкторы

Обычно хеши создаются не конструкторами, а так, как описано в разделе "Конструкции языка Parser".




Open. Открытие или создание

^hashfile::open[имя файла]

Открывает имеющийся на диске файл или создает новый.
Для хранения данных в настоящий момент используются два файла, с суффиксами .dir и .pag.




Поля

В качестве поля хеша выступает ключ, по имени которого можно получить значение:
$my_hash.key

Такая запись возвратит значение, поставленное в соответствие ключу. Если происходит обращение к несуществующему ключу, будет возвращено значение ключа _default, если он задан в хеше.

Для большей взаимозаменяемости таблиц и хешей поле fields хранит ссылку на сам хеш, см. «Использование хеша вместо таблицы».




В выражениях числовое значение хеша

$man[
$.name[Вася]
$.age[22]
$.sex[m]
]
^man._count[]

Вернет: 3.

В выражениях числовое значение хеша равно количеству ключей:
^if($man > 2){больше}

Будет создан файл

$man[
$.name[Вася]
$.age[22]
$.sex[m]
]
$tab_keys[^man._keys[]]
^tab_keys.save[keys.txt]

Будет создан файл keys.txt с такой таблицей:
key
sex
age
name

Новое содержание хеша

$man[
$.name[Вася]
$.age[22]
$.sex[m]
]
$woman[
$.name[Маша]
$.age[20]
$.weight[50]
]
^man.add[$woman]

Новое содержание хеша $man:
$man[
$.name[Маша]
$.age[20]
$.sex[m]
]

с ним значение из хеша

^man.delete[name]

Удалит ключ name и связанное с ним значение из хеша man.

желательно некоторую информацию получить от

Допустим, желательно некоторую информацию получить от посетителя на одной странице сайта, и иметь возможность отобразить ее - на другой странице сайта. Причем необходимо, чтобы посетитель не мог ее ни увидеть ни подделать.

Можно поместить информацию в hashfile, ассоциировав ее со случайной строкой - идентификатором «сеанса общения с посетителем». Идентификатор сеанса общения можно поместить в cookie, данные теперь хранятся на сервере, не видны посетителю и не могут быть им подделаны.

# создаем/открываем файл с информацией
$sessions[^hashfile::open[/sessions]]
^if(!def $cookie:sid){
$cookie:sid[^math:uuid[]]
}
# после этого…

$information_string[произвольное значение]
# …так запоминаем произвольную $information_string под ключом sid на 2 дня
$sid[$cookie:sid]
$sessions.$sid[$.value[$information_string] $.expires(2)]

# …а так можем считать сохраненное ранее значение
# если с момента сохранения прошло меньше 2х дней
$sid[$cookie:sid]
$information_string[$sessions.$sid]

Выведет на

$man[
$.name[Вася]
$.age[22]
$.sex[m]
]
^man.foreach[key;value]{
$key=$value
}[
]

Выведет на экран:
sex=m
age=22
name=Вася

Получится хеш

$man[
$.name[Вася]
$.age[22]
$.sex[m]
]
$woman[
$.name[Маша]
$.age[20]
$.weight[50]
]
$int_hash[^man.intersection[$woman]]

Получится хеш $int_hash:
$int_hash[
$.name[Вася]
$.age[22]
]

^if

^if(^man.intersects[$woman]){
Пересечение есть
}{
Не пересекаются
}

man останется только один ключ

$man[
$.name[Вася]
$.age[22]
$.sex[m]
]
$woman[
$.name[Маша]
$.age[20]
]
^man.sub[$woman]

В результате в хеше $ man останется только один ключ $man.sex со значением m.

Получится хеш

$man[
$.name[Вася]
$.age[22]
$.sex[m]
]
$woman[
$.name[Маша]
$.age[20]
$.weight[50]
]
$union_hash[^man.union[$woman]]

Получится хеш $union_hash:
$union_hash[
$.name[Вася]
$.age[22]
$.sex[m]
$.weight[50]
]

hash of bool[3.1.2]

В БД содержится таблица participants:
participant
Константин
Александр

Выполнение кода…
^connect[строка подключения]{
$participants[^hash::sql{select participant from participants}]
}

…даст хеш такой структуры…
$participants[
$.Константин(1)
$.Александр(1)
]

…из которого можно эффективно извлекать информацию, например, так:
$person[Иван]
$person ^if($participants.$person){
участвует}{не участвует} в мероприятии


hash of hash

В БД содержится таблица hash_table:
pet food aggressive
cat milk very
dog bone never

Выполнение кода…
^connect[строка подключения]{
$hash_of_hash[^hash::sql{select
pet, food, aggressive
from
hash_table
}]
}

…даст хеш такой структуры…
$hash_of_hash[
$.cat[
$.food[milk]
$.aggressive[very]
]
$.dog[
$.food[bone]
$.aggressive[never]
]
]

…из которого можно эффективно извлекать информацию, например, так:
$animal[cat]
$animal
любит $hash_of_hash.$animal.food



Sql. Создание хеша на основе выборки из базы данных

^hash::sql{запрос}
^hash::sql{запрос}[$.limit(n) $.offset(o) $.distinct(d)]


Конструктор создает хеш, в котором имена ключей совпадают со значениями первого столбца выборки. Имена столбцов формируют ключи хеша, а значения столбцов - соответствующие этим ключам значения. Если столбцов больше одного, получается структура hash of hash.
Если же запрос возвращает только один столбец, формируется хеш, где значения столбца формируют ключи хеша, и им ставится в соответствие логическое значение истина. [3.1.2]

Дополнительные параметры конструктора:
$.limit(n) - получить только n записей
$.offset(o) - отбросить первые o записей выборки
$.distinct(1/0) - 1=отбирать записи с уникальным ключом, 0=считать наличие дубликата ошибкой

По-умолчанию, наличие в ключевом столбце одинаковых значений считается ошибкой, если вам необходимо именно отобрать из результата записи с уникальным ключом, задайте опцию $.distinct(1).
Примечание: имейте в виду, что так между клиентом и сервером передаются лишние данные, и, скорее всего, запрос можно изменить, чтобы необходимая уникальность ключа обеспечивалась SQL-сервером. Если вам необходимы данные и в виде таблицы и в виде хеша, подумайте над использованием table::sql в паре с table.hash.




Sub. Вычитание хешей

^хеш.sub[хеш-вычитаемое]

Метод вычитает из хеша другой хеш-вычитаемое, удаляя ключи, общие для обоих хешей.


Union. Объединение хешей

^хеш_a.union[хеш_b]

Метод выполняет объединение двух хешей. Возвращает хеш, содержащий все ключи хеша a и те из b, которых нет в a. Результат необходимо присваивать новому хешу.


Запись

$hashfile.ключ[строка]

Сохраняет на диск ассоциацию между ключом и строкой.

$hashfile.ключ[
$.value[строка]
$.expires(число дней)
]
$hashfile:ключ[
$.value[строка]
$.expires[дата]
]

Такая запись позволяет указать дату устаревания ассоциации. Можно указать число дней или конкретную дату.

Необязательные модификаторы:
$.expires(число дней) - задает число дней (может быть дробным, 1.5=полтора дня), на которое сохраняется пара ключ/строка, 0 дней=навсегда;
$.expires[$date] - задает дату и время, до которой будет храниться ассоциация, здесь $date - переменная типа date.




А что это вообще такое?

Arc. Рисование дуги

^картинка.arc(centerx;center y;width;height;start in degrees;end in degrees;color)
Метод рисует дугу с заданными параметрами. Дуга представляет собой часть эллипса (как частный случай окружности) и задается координатами центра X и Y, шириной, высотой, а также начальным и конечным углом, задаваемым в градусах.


Bar. Рисование закрашенных прямоугольников

^картинка.bar(x0;y0;x1;y1;цветпрямоугольника)

Метод рисует на изображении закрашенный заданным цветом прямоугольник по заданным координатам.


Circle. Рисование неокрашенной окружности

^картинка.circle(centerx;center y;радиус;цвет линии)

Метод рисует окружность заданного радиуса линией заданного цвета относительно центра с координатами X и Y.


Copy. Копирование фрагментов изображений

^картинка.copy[исходное_изображение](x1;y1;ширина1;высота1;x2;y2)^картинка.copy[исходное_изображение](x1;y1;ширина1;высота1;x2;y2;ширина2;высота2;приближение цвета)

Метод копирует фрагмент одного изображения в другое изображение. Это очень удобно использовать в задачах, подобных расставлению значков на карте. В качестве параметров методу передаются:

1.Исходное изображение
2.координаты (X1;Y1) верхнего левого угла копируемого фрагмента
3.ширина и высота копируемого фрагмента
4.координаты (X2;Y2) по которым будет вставлен копируемый фрагмент
5.в качестве необязательных параметров могут быть заданы новая ширина и высота вставляемого фрагмента (в этом случае происходит масштабирование), а также величина, характеризующая точность передачи цвета. Чем она меньше, тем точнее цветопередача, но количество передаваемых цветом уменьшается и наоборот (по умолчанию равна 150)




Create. Создание объекта с заданными размерами

^image::create(размерX; размер Y)
^image::create(размер X; размер Y; цвет фона)

Создает объект класса image размером X на Y. В качестве необязательного параметра можно задать произвольный цвет фона. Если этот параметр пропущен, созданное изображение будет иметь белый цвет фона.


Fill. Закрашивание одноцветной области изображения

^картинка.fill(x;y;цвет)

Метод используется для закрашивания областей изображения, окрашенных в одинаковый цвет, новым цветом. Область закрашивания определяется относительно точки с координатами X и Y.


Font. Загрузка файла шрифта для

^картинка.font[набор_букв;имя_файла_шрифта.gif](ширина_пробела)
^картинка.font[набор_букв;имя_файла_шрифта.gif](ширина_пробела;ширина_символа)


Помимо методов для рисования, Parser также предусматривает возможность нанесения надписей на рисунки. Для реализации этой возможности требуется наличие специальных файлов с изображением шрифтов. Можно либо использовать готовые файлы шрифтов, либо самостоятельно создавать собственные с нужным набором символов.

После загрузки такого файла с помощью метода font набору букв, заданных в параметрах метода, ставятся в соответствие фрагменты изображения из файла. Данный файл должен быть в формате GIF с прозрачным фоном и содержать изображение необходимого набора символов в следующем виде:

Пример файла digits.gif с изображением цифр:
0
1
2
3
4
5
6
7
8
9

Высота каждого символа определяется как отношение высоты рисунка к количеству букв в наборе.
Методу передаются следующие параметры:

Набор букв
- перечень символов, входящих в файл шрифта
Имя и путь к файлу шрифта
Ширина символа пробела (в пикселах)
Ширина символа - необязательный параметр

По умолчанию, при загрузке файла шрифта автоматически измеряется ширина всех его символов и при выводе текста используется пропорциональный (proportional) шрифт. Если задать ширину символа, то шрифт будет моноширинным.

Все символы следует располагать непосредственно у левого края изображения.


Format. Вывод числа в заданном формате

^имя.format[форматнаястрока]

Метод выводит значение переменной в заданном формате (см. Форматные строки).


Gif. Кодирование объектов класса image в формат GIF

^картинка.gif[]
^картинка.gif[имя файла]
[3.1.2]

Используется для кодирования созданных Parser объектов класса image в формат GIF.
Имя файла будет передано посетителю при $response:download.

Внимание: в результате использования этого метода создается новый объект класса file, а не image!

Кроме того, необходимо учитывать тот факт, что цвета выделяются из палитры, и, когда палитра заканчивается, начинается подбор ближайших цветов. В случае создания сложных изображений, особенно с предварительно загруженным фоном, следует иметь в виду последовательность захвата цветов.


Html. Вывод изображения

^картинка.html[]
^картинка.html[хеш]

Создает следующий HTML-тег:


В качестве параметра методу может быть передан хеш, содержащий дополнительные атрибуты изображения, например alt и border, задающие надпись, появляющуюся при наведении курсора и ширину рамки.

Замечание: атрибуты изображения можно переопределять.


Inc, dec, mul, div, mod. Простые операции над числами

^имя.inc[]
^имя.inc(число)

- увеличивает значение переменной на 1 или число
^имя.dec[]
^имя.dec(число)

- уменьшает значение переменной на 1 или число
^имя.mul(число)
- умножает значение переменной на число
^имя.div(число)
- делит значение переменной на число
^имя.mod(число)
- возвращает остаток от деления значения переменной на число




Int, double. Преобразование объектов к числам

^имя.int[]

или
^имя.int(default)
^имя.double[]

или
^имя.double(default)

Преобразуют значение переменной $имя к целому или вещественному числу соответственно, и возвращает это число. При преобразовании вещественного числа к целому производится округление.

Можно задать значение по умолчанию, которое будет получено, если преобразование невозможно. Значение по умолчанию можно использовать при обработке данных, получаемых интерактивно от пользователей. Это позволит избежать появления текстовых значений в математических выражениях при вводе некорректных данных, например, строки вместо ожидаемого числа.

Внимание: пустая строка и строка состоящая только из "white spaces" (символы пробела, табуляция, перевода строки) считается нулем.




Класс image

Класс для работы с графическими изображениями. Объекты класса image бывают двух типов. К первому относятся объекты, созданные на основе существующих изображений в поддерживаемых форматах. Ко второму - объекты, формируемые самим Parser.

Из JPEG файлов можно получить EXIF информацию (http://www.exif.org).

Для представления цветов используется схема RGB, в которой каждый оттенок цвета представлен тремя составляющими компонентами (R-красный, G-зеленый, B-синий). Каждая составляющая может принимать значение от 0x00 до 0xFF (0 - 255 в десятичной системе). Итоговый цвет представляет собой целое число вида 0xRRGGBB, где под каждую составляющую компоненту отведено два разряда в указанной последовательности. Формула для вычисления цвета следующая:

(R*0x100+G)*0x100+B

Так, для белого цвета, у которого все компоненты имеют максимальное значение - FF, данная формула при подстановке дает:

(0xFF*0x100+0xFF)*0x100+0xFF = 0xFFFFFF




Классы int, double

Объектами классов int и double являются целые и вещественные числа, как заданные пользователем, так и полученные в результате вычислений или преобразований. Числа, относящиеся к классу double, имеют представление в формате с плавающей точкой. Диапазон значений зависит от платформы, но, как правило:

для int

от -2147483648
до 2147483647

для double

от 1.7E-308
до 1.7E+308
Класс double обычно имеет 15 значащих цифр и не гарантирует сохранение цифр в последних разрядах. Точное количество значащих цифр зависит от вашей платформы.




Length. Получение длины надписи в пикселах

^картинка.length[текстнадписи]

Метод вычисляет полную длину надписи в пикселах.


Line. Рисование линии на изображении

^картинка.line(x0;y0;x1;y1;цвет)

Метод рисует на изображении линию из точки с координатами (х0:y0) в точку (x1:y1) заданного цвета.


Load. Создание объекта на основе

^image::load[имя_файла.gif]

Создает объект класса image на основе готового фона. Это дает возможность использовать готовые изображения в формате GIF в качестве подложки для рисования, что может использоваться для создания графиков, графических счетчиков и т.п.


Measure. Создание объекта на основе

^image::measure[файл]
^image::measure[имяфайла]

Создает объект класса image, измеряя размеры существующего графического файла или объекта класса file в поддерживаемом формате (сейчас поддерживаются GIF, JPEG и PNG).

Из JPEG файлов также считывается EXIF информация (http://www.exif.org), если она там записана. Большинство современных цифровых фотоаппаратов при записи JPEG файла записывают в него также информацию о снимке, параметрах экспозиции и другую информацию в формате EXIF.

Сама картинка не считывается, основное назначение метода - последующий вызов для созданного объекта метода html.

Параметры:
Файл - объект класса file
Имя файла - имя файла с путем

Примечание: поддерживается EXIF 1.0, считываются теги из IFD0 и SubIFD.




Методы рисования

Данные методы используются только для объектов класса image, созданных с помощью конструкторов create и load. С их помощью можно рисовать линии и различные геометрические фигуры на изображениях и закрашивать области изображений различными цветами. Это дает возможность создавать динамически изменяемые картинки для графиков, графических счетчиков и т.п.

Отсчет координат для графических объектов ведется с верхнего левого угла, точка с координатами (0:0).




Описание формата файла, описывающего кодировку

Данные в формате tab-delimited со следующими столбцами:

char - символ, или его код, заданный в десятичной или шестнадцатеричной форме
(0xHH) в той кодировке, которую определяет этот файл.

white-space, digit, hex-digit, letter, word - набор флажков, задающих класс этого символа. Пустое содержимое означает непринадлежность символа к этому классу, непустое [например, 'x'] - принадлежность.

Подробнее о символьных классах см. описание регулярных выражений в литературе.

lowercase - если символ имеет пару в нижнем регистре, то символ или код парного символа. Скажем, у буквы 'W' есть парная 'w'. Используется в регулярных выражениях для поиска, нечувствительного к регистру символов, а также в методах lower и upper класса string.

unicode1 - основной Unicode код символа. Если совпадает с кодом символа, то можно не указывать. Скажем, у буквы 'W' он совпадает, а у буквы 'Я' - нет.

unicode2 - дополнительный Unicode символа, если имеется.




Pixel. Работа с точками изображения

^картинка.pixel(x;y)

Выдает цвет указанной точки изображения. Если координаты попадают за пределы изображения, выдает -1.

^картинка.pixel(x;y;цвет)

Задает цвет указанной точки.




Поля

$картинка.src




Polybar. Рисование окрашенных

^картинка.polybar(цветмногоугольника)[таблица с координатами узлов]

Метод рисует многоугольник заданного цвета по координатам узлов, задаваемым в таблице. Последний узел автоматически соединяется с первым.


Polygon. Рисование неокрашенных

^картинка.polygon(цветлинии)[таблица с координатами узлов]

Метод рисует линией заданного цвета многоугольник по координатам узлов, задаваемым в таблице. Последний узел автоматически соединяется с первым.


Polyline. Рисование ломаных линий по координатам узлов

^картинка.polyline(цвет)[таблицас координатами точек]

Метод рисует линию по координатам узлов, задаваемым в таблице. Он используется для создания ломаных линий.


В браузере будет выведен черный

$square[^image::create(100;100;0x000000)]
^square.arc(50;50;40;40;0;90;0xFFFFFF)
$response:body[^square.gif[]]

В браузере будет выведен черный квадрат с дугой в четверть (от 0 до 90 градусов) окружности радиусом 40 пикселов.

В браузере будет выведен черный

$square[^image::create(100;100;0x000000)]
^square.bar(5;40;95;60;0xFFFFFF)
$response:body[^square.gif[]]

В браузере будет выведен черный квадрат размером 100 на 100 пикселов, внутри которого находится белый прямоугольник 90 x 20 пикселов, нарисованный по заданным координатам.

В браузере будет выведен черный

$square[^image::create(100;100;0x000000)]
^square.circle(50;50;10;0xFFFFFF)
$response:body[^square.gif[]]

В браузере будет выведен черный квадрат с окружностью радиусом в десять пикселов, нарисованной линией белого цвета с центром в точке (50;50).

В данном примере мы создаем

$mygif[^image::load[test.gif]]

$resample_width($mygif.width*2)
$resample_height($mygif.height*2)

$mygif_new[^image::create($resample_width;$resample_height)]
^mygif_new.copy[$mygif](0;0;20;30;0;0;$mygif_new.width;$mygif_new.height)

$response:body[^mygif_new.gif[]]

В данном примере мы создаем два объекта класса image. Первый создан на основе существующего GIF файла. Второй - вдвое больший по размеру, чем первый, создается самим Parser, после чего в него мы копируем фрагмент первого размером 20х30 и «растягиваем» этот фрагмент на всю ширину и высоту второго рисунка. Последняя строчка кода выводит увеличенный фрагмент на экран. Данный подход можно применять только для изображений, которые не требуется выводить с хорошим качеством.

Будет создан объект square класса

$square[^image::create(100;100;0x000000)]

Будет создан объект square класса image размером 100х100 c черным цветом фона.

в этом файле. Если снимок

$photo[^image::measure[photo.jpg]]
Имя файла: $photo.src

Ширина изображения в пикселах: $photo.width

Высота изображения в пикселах: $photo.height

$date_time_original[$photo.exif.DateTimeOriginal]
^if(def $date_time_original){
Снимок сделан ^date_time_original.sql-string[]

}

Будет выведено имя файла, а также ширина и высота изображения, хранящегося в этом файле. Если снимок был сделан цифровым фотоаппаратом, вероятно, будет выведена дата и время съемки.

В браузере будет выведен квадрат

$square[^image::create(100;100;0x000000)]
^square.line(0;0;100;100;0xFFFFFF)
^square.fill(10;0;0xFFFF00)
$response:body[^square.gif[]]

В браузере будет выведен квадрат размером 100 на 100 пикселов, перечеркнутый по диагонали белой линией. Нижняя половина квадрата черная, а верхняя закрашена желтым цветом.

В данном случае будет загружен

$square[^image::create(100;100;0x00FF00)]
^square.font[0123456789;digits.gif](0)

В данном случае будет загружен файл, содержащий изображения цифр от 0 до 9, и набору цифр от 0 до 9 будет поставлено в соответствие их графическое изображение. После того, как определен шрифт для нанесения надписи, можно использовать метод text для нанесения надписей.

В браузере будет выведен черный

$square[^image::create(100;100;0x000000)]
$response:body[^square.gif[]]

В браузере будет выведен черный квадрат размером 100 на 100 пикселов.

В браузере будет выведена картинка

$photo[^image::measure[myphoto.jpg]]
^photo.html[
$.border[0]
$.alt[Это я в молодости…]
]

В браузере будет выведена картинка из переменной $photo. При наведении курсора будет появляться надпись: это я в молодости…

В результате будет вычислена длина

$square[^image::create(100;100;0x00FF00)]
^square.font[0123456789;digits.gif](0)
^square.length[128500]

В результате будет вычислена длина надписи «128500» в пикселах с учетом пробелов.

В браузере будет выведен черный

$square[^image::create(100;100;0x000000)]
^square.line(0;0;100;100;0xFFFFFF)
$response:body[^square.gif[]]

В браузере будет выведен черный квадрат размером 100 на 100 пикселов перечеркнутый по диагонали белой линией.

Будет создан объект класса image

$background[^image::load[counter_background.gif]]

Будет создан объект класса image на основе готового изображения в формате GIF. Этот объект может впоследствии использоваться для подложки в методах рисования.

В браузере будет выведен равнобедренный

$rectangle_coordinates[^table::create{x y
0 0
50 100
100 0
}]

$square[^image::create(100;100;0x000000)]
^square.polybar(0x00FF00)[$rectangle_coordinates]
$response:body[^square.gif[]]

В браузере будет выведен равнобедренный треугольник зеленого цвета на черном фоне. В таблице заданы координаты вершин треугольника.

В браузере будет выведен равнобедренный

$rectangle_coordinates[^table::create{x y
0 0
50 100
100 0
}]

$square[^image::create(100;100;0x000000)]
^square.polygon(0x00FF00)[$rectangle_coordinates]
$response:body[^square.gif[]]

В браузере будет выведен равнобедренный треугольник, нарисованный линией зеленого цвета на черном фоне. В таблице заданы координаты вершин треугольника.

В браузере будет выведена буква

$table_with_coordinates[^table::create{x y
10 0
10 100
20 100
20 50
50 50
50 40
20 40
20 10
60 10
65 15
65 0
10 0
}]
$square[^image::create(100;100;0xFFFFFF)]

$square.line-style[*** ]
$square.line-width(2)

^square.polyline(0xFF00FF)[$table_with_coordinates]

$file_withgif[^square.gif[]]
^file_withgif.save[binary;letter_F.gif]

$letter_F[^image::load[letter_F.gif]]
^letter_F.html[]

В браузере будет выведена буква F, нарисованная пунктирной линией на белом фоне. В рабочем каталоге будет создан файл letter.gif. В этом примере используются объекты класса image двух различных типов. В таблице задаются координаты точек ломанной линии. Затем на созданном с помощью конструктора create фоне рисуется линия по указанным координатам узлов. Созданный объект класса image кодируется в формат GIF. Полученный в результате этого объект класса file сохраняется на диск. Затем создается новый объект класса image на основе сохраненного файла. Этот объект выводится на экран браузера методом html.

В браузере будет выведен черный

$square[^image::create(100;100;0x000000)]
^square.rectangle(5;40;95;60;0xFFFFFF)
$response:body[^square.gif[]]

В браузере будет выведен черный квадрат размером 100 на 100 пикселов, внутри которого находится прямоугольник 90 x 20 пикселов, нарисованный линией белого цвета по заданным координатам.

В браузере будет выведен черный

$paint_nodes[^table::create{x y
10 20
90 20
90 80
10 80
}]

$square[^image::create(100;100;0x000000)]
^square.line(0;0;100;100;0xFFFFFF)
^square.line(100;0;0;100;0xFFFFFF)

^square.replace(0x000000;0xFF00FF)[$paint_nodes]
$response:body[^square.gif[]]

В браузере будет выведен черный квадрат, перечеркнутый по диагонали белыми линиями, со вписаным в него розовым прямоугольником. Поскольку в методе replace задана замена на розовый цвет только для черного цвета, белые линии не перекрасились.

В браузере будет выведен черный

$square[^image::create(100;100;0x000000)]
^square.sector(50;50;40;40;0;90;0xFFFFFF)
$response:body[^square.gif[]]

В браузере будет выведен черный квадрат с сектором в четверть (от 0 до 90 градусов) окружности радиусом 40 пикселов. Сектор нарисован линией белого цвета.

Для методов рисования будет использоваться

$картинка.line-style[*** ]
$картинка.line-width(2)

Для методов рисования будет использоваться пунктирная линия вида:
*** *** *** *** ***
толщиной в два пиксела.

В браузере будет выведен зеленый

$square[^image::create(100;100;0x00FF00)]
^square.font[0123456789;digits.gif](0)

^square.text(5;5)[128500]
$response:body[^square.gif[]]

В браузере будет выведен зеленый квадрат с надписью «128500» левая верхняя точка которой находится в точке с координатами (5;5).

и эквивалентен записи

$var(5)
^var.inc(7)
^var.dec(3)
^var.div(4)
^var.mul(2)
$var

Пример возвратит 4.5 и эквивалентен записи $var((5+7-3)/4*2).

Вернет количество записей

^connect[строка подключения]{
^int:sql{select count(*) from news}
}

Вернет количество записей в таблице news.

Пример работы с EXIF информацией

$image[^image::measure[jpg/DSC00003.JPG]]
$exif[$image.
exif]
^if($exif){
Производитель фотоаппарата, модель: $exif.Make $exif.Model

Время съемки: ^exif.DateTimeOriginal.sql-string[]

Выдержка: $exif.ExposureTime секунды

Диафрагма: F$exif.FNumber

Использовалась вспышка: ^if(def $exif.Flash){^if($exif.Flash){да;нет};неизвестно}

}{
нет EXIF информации

}




Пример создания тега IMG с указанием размеров изображения

$photo[^image::measure[photo.png]]
^photo.html[]

Будет создан объект photo класса image, на основе готового графического изображения в формате PNG, и выдан тег IMG, ссылающийся на данный файл, с указанием width и height.


^var.format

$var(15.67678678)
^var.format[%.2f]

Возвратит: 15.68

$var(0x123)
^var.format[0x%04X]

Возвратит: 0x0123

Выведет число 1024, поскольку объект

$str[Штука]
^str.int(1024)

Выведет число 1024, поскольку объект str нельзя преобразовать к классу int.

$double(1.5)
^double.int[]

Выведет число 2, поскольку произведено автоматическое округление.

Rectangle. Рисование незакрашенный прямоугольников

^картинка.rectangle(x0;y0;x1;y1;цветлинии)

Метод рисует на изображении незакрашенный прямоугольник по заданным координатам с заданным цветом линии.


Replace. Замена цвета в области, заданной таблицей координат

^картинка.replace(старыйцвет;новый цвет)[таблица с координатами точек]

Метод используется для замены одного цвета другим в области изображения, заданной с помощью таблицы координат.


Sector. Рисование сектора

^картинка.sector(centerx;center y;width;height;start in degrees;end in degrees;color)
Метод рисует сектор с заданными параметрами линией заданного цвета. Параметры метода аналогичны методу arc.


Sql. Получение числа из базы данных

^int:sql{запрос}
^int:sql{запрос}[$.limit(1) $.offset(o) $.default{код}]
^double:sql{запрос}
^double:sql{запрос}[$.limit(1) $.offset(o) $.default{код}]

Возвращает число, полученное в результате SQL-запроса к серверу баз данных. Запрос должен возвращать значение из одного столбца одной строки.

Запрос
- запрос к базе данных, написанный на языке SQL
$.offset(o)
- отбросить первые o записей выборки
$.default{код}
- если результат SQL-запроса не число, то метод возвратит результат выполнения кода
Для работы этого метода необходимо установленное соединение с сервером базы данных (см. оператор connect).


Text. Нанесение надписей на изображение

^картинка.text(x;y)[текстнадписи]

Метод выводит заданную надпись по указанным координатам (X;Y), используя файл шрифта, предварительно загруженный методом font


Тип и ширина линий

$картинка.line-style[типлинии]
$картинка.line-width(толщина линии)

Перед вызовом любых методов рисования можно задавать тип и толщину используемых линий.
Тип линии задается строкой, где пробелы означают отсутствие точек в линии, а любые другие символы - наличие.


Установка и настройка Parser

Parser3 доступен в нескольких вариантах:
·CGI скрипт (и интепретатор),
·модуль к веб-серверу Apache 1.3,
·ISAPI расширение веб-сервера Microsoft Internet Information Server 4.0 или новее.

Дополнительно можно установить драйверы для различных SQL-серверов (сейчас доступны для MySQL, PgSQL, Oracle, ODBC).

Описание каталогов и файлов :
parser3[.exe] - CGI скрипт (и интерпретатор)
ApacheModuleParser3.dll - модуль к веб-серверу Apache 1.3
parser3isapi.dll - ISAPI расширение веб-сервера IIS 4.0 или новее

auto.p.dist - пример Конфигурационного файла

parser3.charsets/ - каталог с файлами таблиц кодировок:

koi8-r.cfg

- Cyrillic [KOI8-R]

windows-1250.cfg

- Central european[windows-1250]

windows-1251.cfg

- Cyrillic[windows-1251]

windows-1257.cfg

- Baltic[windows-1257]
Поскольку исходные коды являются открытыми, вы можете сами собрать Parser (см. Сборка Parser их исходных кодов) и написать свой SQL-драйвер.
Доступны скомпилированные версии Parser и его SQL-драйверов под ряд платформ (см. http://parser.ru/download/).
Внимание: в целях безопасности они скомпилированы так, что могут читать и исполнять только файлы, принадлежащие тому же пользователю/группе пользователей, от имени которых работает сам Parser.

Как подключаются конфигурационные файлы?

Для CGI скрипта (parser3[.exe]):
конфигурационный файл считывается из файла, заданного переменной окружения CGI_PARSER_CONFIG,
Если переменная не задана, ищется в том же каталоге, где расположен сам CGI скрипт.

Для модуля к Apache путь к конфигурационному файлу задается директивой:
#can be in .htaccess
ParserConfig полный_путь

Для ISAPI расширения (parser3isapi.dll):
конфигурационный файл auto.p ищется в том же каталоге, где расположен сам файл.




Установка Parser на веб-сервер Apache 1.3, модуль сервера

Для установки Parser необходимо внести изменения в основной конфигурационный файл веб-сервера, или, если у вас нет к нему доступа, необходима возможность использовать .htaccess файлы.

По-умолчанию, в установке Apache возможность использования .htaccess отключена.
Если она вам необходима, разрешите их использовать (по крайней мере, задавать FileInfo). Для чего в основном конфигурационном файле веб-сервера (обычно httpd.conf) в секцию вашего вашего сайта, или вне ее - для всех сайтов, добавьте директивы:
<Directory /путь/к/вашему/веб/пространству>
AllowOverride FileInfo


Parser3 самостоятельно выполняет необходимые перекодирования, так что для Русского Apache добавьте в основной конфигурационный файл веб-сервера (обычно httpd.conf) строку:
CharsetDisable On
запрещающую использование возможностей перекодирования Русского Apache для вашего сервера.
Если возможности изменить основной конфигурационный файл веб-сервера у вас нет, добавьте эту строку в .htaccess файл.

Под UNIX:
Необходимо собрать Parser из исходных кодов, задав ключ --with-apache13 у скрипта configure, при make на экране появится инструкция по дальнейшей сборке Apache из его исходных кодов.
Внимание: на некоторых системах стандартный make не работает с make-файлами Parser3, воспользуйтесь GNU вариантом: gmake.

Под Win32:
Поместите файлы с исполняемым кодом модуля Parser (в текущей версии, ApacheModuleParser3.dll) в произвольный каталог.
Если вы используете версию Parser с поддержкой XML, в каталог, указанный в переменной окружения PATH (например, C:\WinNT), распакуйте XML библиотеки.
Добавьте в файл httpd.conf после имеющихся строк LoadModule:
# динамическая загрузка модуля
LoadModule parser3_module x:\path\to\ApacheModuleParser3.dll
Внимание: если необходимо, поместите сопутствующие .dll файлы в тот же каталог.

А после имеющихся строк AddModule (если не имеются, не добавляйте):
# добавление модуля к списку активных модулей
AddModule mod_parser3.c
Внимание: до 3.1.1 версии Parser: AddModule mod_parser3.C (на конце .C большая)

Добавьте в файл .htaccess вашего сайта (или в httpd.conf в секцию вашего сайта, или вне ее - для всех сайтов) блоки:
# назначение обработчиком .html страниц
AddHandler parser3-handler html

# задание Конфигурационного файла
ParserConfig x:\path\to\parser3\config\auto.p

# запрет на доступ к .p файлам, в основном, к auto.p
<
Files ~ "\.p$">
Order allow,deny
Deny from all





Установка Parser на веб-сервер Apache, CGI скрипт

Для установки Parser необходимо внести изменения в основной конфигурационный файл веб-сервера, или, если у вас нет к нему доступа, необходима возможность использовать .htaccess файлы.

По-умолчанию, в установке Apache возможность использования .htaccess отключена.
Если она вам необходима, разрешите их использовать (по крайней мере, задавать FileInfo). Для чего в основном конфигурационном файле веб-сервера (обычно httpd.conf) в секцию вашего сайта, или вне ее - для всех сайтов, добавьте директивы:
<Directory /путь/к/вашему/веб/пространству>
AllowOverride FileInfo


Parser3 самостоятельно выполняет необходимые перекодирования, так что для Русского Apache добавьте в основной конфигурационный файл веб-сервера (обычно httpd.conf) строку:
CharsetDisable On
запрещающую использование возможностей перекодирования Русского Apache для вашего сервера.
Если возможности изменить основной конфигурационный файл веб-сервера у вас нет, добавьте эту строку в .htaccess файл.

Поместите файл с исполняемым кодом Parser (в текущей версии, parser3) в каталог для CGI-скриптов.
Под Win32: если вы используете версию Parser с поддержкой XML, в этот же каталог распакуйте XML библиотеки.

Добавьте в файл .htaccess вашего сайта (или в httpd.conf в секцию вашего сайта, или вне ее - для всех сайтов) блоки:
# назначение обработчиком .html страниц
AddHandler parser3-handler html
Action parser3-handler /cgi-bin/parser3

# запрет на доступ к .p файлам, в основном, к auto.p
<
Files ~ "\.p$">
Order allow,deny
Deny from all


Если вас не устраивает расположение конфигурационного файла по умолчанию (см. Установка и настройка Parser), вы можете задать его явно:

# задание переменной окружения с путем к auto.p
SetEnv CGI_PARSER_CONFIG /путь/к/файлу/auto.p

Замечание: для этого необходим модуль mod_env, который по умолчанию установлен.

Об ошибках Parser делает записи в журнал ошибок parser3.log, который, по умолчанию, расположен в том же каталоге, где и CGI-скрипт Parser. Если у Parser нет возможности сделать запись в данный файл, об ошибке будет сообщено в стандартный поток ошибок, и запись об ошибке попадет в журнал ошибок веб-сервера. Если вас не устраивает расположение журнала ошибок parser3.log, вы можете задать его явно:

# задание переменной окружения с путем к parser3.log
SetEnv CGI_PARSER_LOG /путь/к/файлу/parser3.log

Замечание: для этого необходим модуль mod_env, который по умолчанию установлен.




Установка Parser на веб-сервер IIS 5.0 или новее

Поместите файлы с исполняемым кодом модуля Parser (в текущей версии, parser3isapi.dll) в произвольный каталог.
Если вы используете версию Parser с поддержкой XML, в каталог, указанный в переменной окружения PATH (например, C:\WinNT), распакуйте XML библиотеки.

После чего назначьте Parser обработчиком .html страниц:
1.Запустите Management Console, нажмите на правую кнопку мыши на названии вашего веб-сервера и выберите Properties.
2.Перейдите на вкладку Home directory и в разделе Application settings нажмите на кнопку Configuration…
3.В появившемся окне нажмите на кнопку Add.
4.В поле Executable введите полный путь к файлу parser3.exe или parser3isapi.dll.
5.В поле Extension введите строку .html.
6.Включите опцию Check that file exists.
7.Нажмите на кнопку OK.




А что это вообще такое?

Класс junction

Класс предназначен для хранения кода и контекста его выполнения.
При обращении к переменным, хранящим в себе junction, Parser выполняет код в сохраненном контексте.

Значение типа junction появляется в переменной…

…при присваивании ей кода:
$junction{Код,присваиваемый переменной: ^do_something[]}

…при передачи кода параметром:
@somewhere[]
^method{
Код, передаваемый параметром: ^do_something_else[]}

@method[parameter]
#здесь в $parameter придет junction

…при обращении к имени метода класса:
$action[$user:edit]
#$action[
$user:delete]
^action[параметр]

Здесь $action хранит ссылку на метод и его класс, вызов action теперь аналогичен вызову ^edit[параметр].

…при обращении к имени метода объекта:
$action[$person.show_info]
^action[full]

Здесь $action хранит ссылку на метод и его объект, вызов action теперь аналогичен вызову ^person.show_info[параметры].


junction выражений и кода

@possible_reminder[age;have_passport]
^myif($age>=16 && !$have_passport){
Тебе уже $age лет, пора сходить в милицию.
}

@myif[condition;action][age]
$age(11)
^if($condition){
$action
}


Напоминание: параметр с выражением, это код, вычисляющий выражение, он выполняется - вычисляется выражение - при каждом обращении к параметру внутри вызова.

Здесь оператору myif передан код, печатающий, среди прочего, $age. Выполнение проверки и кода оператор производит в сохраненном (внутри $condition и $action) контексте, поэтому наличие в myif локальной переменной age и ее значение никак не влияет на то, что будет проверено и что напечатано.

Пример проверки наличия метода

^if($some_method is junction){
^some_method[параметр]
}
{
нет метода
}

Метод some_method, будет вызван только, если определен.




А что это вообще такое?

Что делать дальше: на этом создание

Удачи!




Что надо запомнить: Parser использует

Синтаксис работы с объектами:

$переменная[значение]
- задаем значение
$переменная
- получаем значение
$переменная
[^имя_класса::конструктор[параметры]]

- создаем объекта класса имя_класса и присваиваем его переменной
$переменная.имя_поля
- получаем поле самого объекта, хранящегося в переменной
^переменная.метод[]
- вызываем действие (метод класса, к которому принадлежит объект, хранящийся в переменной)




Что надо запомнить:

·наряду с методами объектов, создаваемых с помощью конструкторов класса, существуют статические методы. Вы можете непосредственно обращаться к этим методам без предварительного использования конструктора для создания объекта
·в циклах for можно обращаться к счетчику как к переменной по имени и получать его текущее значение

Поскольку код становится все объемнее, неплохо бы начать снабжать его комментариями, чтобы потом было легче разбираться. В Parser комментариями считается любая строка, начинающаяся со знака #. До сих пор мы не пользовались этим, но в дальнейшем нам пригодится комментирование кода. Следующая строка - пример комментария:

# весь этот текст Parser проигнорирует - это комментарий !!!!

Обязательно комментируйте свой код! В идеале он должен быть самодокументирующимся, а человек, читающий его, должен сразу же понимать о чем идет речь, что из чего следует и т.д. Если этого не сделать, то спустя какое-то время вспомнить что делает та или иная функция станет очень сложно даже вам самим, не говоря уже про остальных. Помните об этом!


Что узнали:

·механизм взаимодействия Parser с СУБД MySQL
·как осуществлять различные SQL-запросы к БД (статический метод sql класса void и конструктор sql класса table)
·оператор untaint




Что узнали:

·класс MAIN
·создание пользовательского класса
·как передавать параметры методам




Что узнали:

·класс xdoc
·как загружать XML, делать XSLT преобразования, выводить объекты класса xdoc в виде HTML




Что узнали:

·файл auto.p может содержаться не только в корневом каталоге сайта, но и в любом другом, при этом функции, в нем определенные, явно доступны только внутри этого каталога
·переменная-хеш - это массив, нужный для построения ассоциативной связи одних объектов с другими. В нашем случае объектами выступали строки
·статический метод calendar создает таблицу с календарем на текущий месяц
·цикл for позволяет последовательно выполнить определенные действия заданное количество раз




Циклы

Цикл - процесс многократного выполнения некоторой последовательности действий.

В Parser существует два оператора циклов: for, в котором количество повторов тела цикла ограничивается заданными значениями счетчика, и while, где количество повторов зависит от выполнения условия. Для того, чтобы избежать зацикливания, в Parser встроен механизм обнаружения бесконечных циклов. Бесконечным считается цикл, тело которого выполняется более 10000 раз.




в Parser принадлежит конкретному классу,

Любой объект в Parser принадлежит конкретному классу, характеризуется полями и методами именно этого класса. Чтобы он появился, его нужно создать. Делает это конструктор данного класса. Разберитесь с этой терминологией, это основа.

Отвлеклись? Продолжим. Переменной $sections мы присвоили вот что:

^table::load[sections.cfg]

Буквально это означает следующее: мы создали объект класса table при помощи конструктора load. Общее правило для создания объекта записывается так:

^имя_класса::конструктор[параметры_конструктора]

Подробности в разделе «Передача параметров».

В качестве параметра конструктору мы передали имя файла с таблицей и путь к нему.

Теперь переменная $sections содержит таблицу с разделами нашего сайта. Parser считает ее объектом класса table и точно знает, какие действия с ней можно выполнить. Пока нам понадобится только один метод этого класса - menu, который последовательно перебирает все строки таблицы. Также нам потребуются значения из полей самой таблицы. Синтаксис вызова методов объекта:

^объект.метод_класса[параметры]

Получение значений полей объекта (мы ведь имеем дело с вполне определенной таблицей с заданными нами же полями):

$объект.имя_поля

Знания, полученные выше, теперь позволяют без труда разобраться в последней части нашего кода:



^sections.menu{

}


$sections.name


Мы формируем HTML-таблицу, в каждый столбец которой помещаем значения, содержащиеся в полях нашей таблицы $sections: uri - адрес и name - имя. При помощи метода menu мы автоматически перебираем все строки таблицы. Таким образом, даже если у нас будет несколько десятков разделов, ни один из них не будет потерян или пропущен. Мы можем свободно добавлять разделы, удалять их и даже менять местами. Изменения вносятся только в файл sections.cfg. Логика работы не нарушится. Все просто и красиво.

Давайте подведем итоги первого урока.

Меню навигации

Давайте начнем с самого начала. Итак, вы хотите сделать сайт (узел, сервер). Первым делом, необходимо уяснить, каким образом на сайте будет упорядочена та или иная информация. Сколько будет категорий, подразделов т.д. Все эти вопросы возникают на первом этапе - "Организация сайта".

А какой должна быть навигация сайта? Требований к хорошей навигации много. Она должна быть понятна, легко узнаваема, единообразна, удобна в использовании, быстро загружаться, давать четкое понятие о текущем местоположении. При этом на сайте не должно возникать 404-й ошибки, т.е. все ссылки должны работать. Если у вас есть опыт создания сайтов, то вам, скорее всего, приходилось сталкиваться с проблемой создания грамотной навигации.

Не правда ли, хочется иметь какое-то решение, которое всегда будет под рукой и позволит автоматизировать весь этот процесс? Что-то такое, что даст возможность единственный раз написать код и потом, в одном месте, дописывать столько разделов, сколько нужно?

Создание меню, которое ориентирует пользователя на сайте, не дает ему заблудиться - вот задача, с которой нам хочется начать повествование о Parser. Почему именно это? Прежде всего потому, что большое количество тегов:



трудно контролировать. А если вам понадобится добавить еще один раздел? Придется в каждую страницу вносить изменения, а человеку свойственно делать ошибки. При этом отнюдь не исключено, что после такой «модернизации» ваш ресурс в ответ на запросы пользователей сообщит о том что «данная страница не найдена». Вот где проблема, которую с помощью Parser можно решить очень легко.

Решение следующее. Мы создаем некую функцию на Parser, которая будет генерировать нужный нам фрагмент HTML-кода. В терминологии Parser функции называются методами. В тех местах, где этот код понадобится, будем просто давать указание «Вставить меню навигации» и сразу же будет создана страница, содержащая меню. Для этого сделаем несколько простых шагов:

1. Всю информацию о наших ссылках будем хранить в одном файле, что позволит впоследствии вносить необходимые изменения только в нем. В корневом каталоге будущего сайта создаем файл sections.cfg, в который помещаем следующую информацию:

section_id

name
uri

1

Главная
/

2

Новости
/news/

3

Контакты
/contacts/

4

Цены
/price/

5

Ваше мнение
/gbook/


Здесь используется так называемый формат tab-delimited. Столбцы разделяются знаком табуляции, а строки - переводом каретки. При копировании этой таблицы в текстовый редактор данное форматирование будет создано автоматически, но если вы будете создавать таблицу вручную, необходимо это учитывать. Для таблиц ВСЕГДА применяется формат tab-delimited.

2. В том же каталоге, где и sections.cfg, создаем файл auto.p
В нем мы будем хранить все те кирпичики, из которых впоследствии Parser соберет наш сайт. AUTO означает, что все эти кирпичики будут всегда доступны для Parser в нужный момент, а расширение ".p", как вы, наверное, догадались, это… правильно! Он самый!

3. В файл auto.p вставим следующий код:

@navigation[]
$sections[^
table::load[sections.cfg]]


^sections.menu{

}


$sections.name


Данные из этого файла и будут служить основой для нашего навигационного меню.

Вот и все, подготовительные работы закончены. Теперь открываем код страницы, где все это должно появиться (например, index.html), и говорим: «Вставить меню навигации». На Parser это называется «вызов метода» и пишется так:

^navigation[]

Осталось только открыть в браузере файл, в который мы вставили вызов метода и посмотреть на готовое меню навигации. Теперь в любом месте на любой странице мы можем написать заветное ^navigation[], и Parser вставит туда наше меню. Страница будет сформирована «на лету». Что хотели, то и получили.

Если у вас дела обстоят именно так, то поздравляем - вы вступили в мир динамических сайтов. Очень скоро вы также запросто будете использовать базы данных для формирования страниц и делать многое другое.

Однако не будем радоваться раньше времени. Давайте разберемся, что же мы сделали, чтобы добиться такого результата. Взгляните на код в auto.p. Если кажется, что все непонятно, не надо бежать прочь. Уверяем, через несколько минут все встанет на свои места. Итак, посмотрим на первую строчку:

@navigation[]

Она аналогична строке ^navigation[], которую мы вставили в текст страницы для создания меню. Различие только в первом символе: ^ и @. Однако логический смысл этого выражения совершенно иной - здесь мы определяем метод, который вызовем позже. Символ @ (собака) в первой колонке строки в Parser означает, что мы хотим описать некоторый блок, которым воспользуемся в дальнейшем. Следующее слово определяет имя нашего метода: navigation. Это только наше решение, как ее назвать. Вполне допустимы имена: а_ну_ка_вставь_меню_быстро. Но читаться такая программа будет хуже, впрочем, кому как понятнее, можете назвать и так.

Жизненно необходимо давать простые, понятные имена. Они должны точно соответствовать тому, что именуемый объект будет хранить и делать. Сохраните нервы и время себе и всем тем, кому придется разбираться в ваших текстах, отнеситесь, пожалуйста, к именам внимательно. Имена могут быть русские или латинские, главное соблюдать единообразие: или все по-русски, или по-английски.

Идем дальше.

$sections[^table::load[sections.cfg]]

Это ключевая строка нашего кода. Она достаточно большая, поэтому давайте разберем ее по частям.

Строка начинается символом $ (рубль) и следующим сразу за ним именем sections. Так в Parser обозначаются переменные. Это надо запомнить. Все просто: видим в тексте $var - имеем дело с переменной var. Переменная может содержать любые данные: числа, строки, таблицы, файлы, рисунки и даже часть кода. Присвоение переменной $parser_home_url значения www.parser3.ru на Parser выглядит так: $parser_home_url[www.parser3.ru]. После этого мы можем обратиться к переменной по имени, т.е. написать $parser_home_url и получить значение www.parser3.ru.

Еще раз тоже самое:

$var[…]
- присваиваем
$var
- получаем
Подробности в разделе «Переменные».

В нашем случае переменная $sections будет хранить таблицу из файла sections.cfg.

Любую таблицу Parser рассматривает как самостоятельный объект, с которым он умеет производить только вполне определенные действия, например, добавить или удалить из нее строку. Поскольку переменная может хранить любые данные, необходимо указать, что присвоенное нами переменной значение является именно таблицей.

Лирическое отступление.
Пример из жизни. Всю автомобильную технику можно грубо разделить на несколько классов: легковые автомашины, грузовики, трактора и гусеничная техника. Любой автомобиль является объектом одного из этих классов. Вы легко можете определить, к какому классу относится автомобиль, поскольку их всех объединяют общие характеристики, такие как вес, масса перевозимого груза и т.д. Любой автомобиль может совершать действия: двигаться, стоять или ломаться. Каждый из автомобилей обладает своими собственными свойствами. И самое главное, автомобиль не может появиться сам собой, его нужно создать. Когда конструктор придумывает новую модель автомобиля, он точно знает, автомобиль какого класса он создает, какими свойствами будет наделено его творение и что оно сможет делать. Также и в Parser: каждый объект относится к определенному классу, объект класса создается конструктором этого класса и наделен характеристиками (полями) и методами (действиями), общими для всех подобных объектов.

Меню навигации и структура страниц

Предыдущий урок мы закончили тем, что определили недостатки в реализации меню. Давайте займемся их устранением. Наше меню выводит лишнюю ссылку на текущую страницу, что нисколько не украшает будущий сайт. Чтобы этого избежать, необходимо проверить, не является ли раздел, на который мы выводим ссылку, текущим. Иными словами, нам нужно сравнить URI раздела, на который собираемся ставить ссылку, с текущим URI. В случае если они совпадают, ссылку на раздел ставить не надо. Дополнительно для удобства пользователей мы изменим в меню навигации цвет столбца текущего раздела.

Открываем файл auto.p и меняем его содержимое на:

@navigation[]
$sections[^
table::load[/sections.cfg]]


^sections.menu{
^navigation_cell[]
}



@navigation_cell[]
$cell_width[^
eval(100\$sections)%]
^if($sections.
uri eq $request:uri){

$sections.name

}{

$sections.name

}

Что изменилось? На первый взгляд не так уж и много, но функциональность нашего модуля существенно возросла. Мы описали еще один метод - navigation_cell, который вызывается из метода navigation. В нем появилась новая структура:

^if(условие){код если условие "истина"}{код если условие "ложь"}

Что она делает, понять не сложно. В круглых скобках задается условие, в зависимости от того, какое значение возвращает условие, "ложь" или "истина", можно получить разный результат. Также, если в условии записано выражение, значение которого равно нулю, то результат - "ложь", иначе - "истина". Мы используем оператор if для того, чтобы в одном случае поставить ссылку на раздел, а другом нет. Осталось только разобраться с условием. Будем сравнивать на равенство две текстовых строки, в одной из которых - значение URI раздела из таблицы sections, в другой - текущий URI ($request:uri возвращает строку, содержащую URI текущей страницы). Тут возникает вопрос о том, какие же строки равны между собой? Несомненно, только те, которые полностью совпадают и по длине, и по символьному содержанию.

Для сравнения двух строк в Parser предусмотрены следующие операторы:

eq - строки равны (equal): parser eq parser
ne - строки не равны (not equal): parser ne parser3
lt
- первая строка меньше второй (less than): parser lt parser3
gt - первая строка больше второй (greater than): parser3 gt parser
le
- первая строка меньше или равна второй (less or equal)
ge - первая строка больше или равна второй (greater or equal)

С условием разобрались: если $sections.uri и $request:uri совпадают, ссылку не ставим (а заодно красим столбец в другой цвет - подумаем о наших пользователях, так им будет удобнее), если нет - ставим.

Идем дальше. Меню из первого урока выводило столбцы разной ширины. Ничего страшного, но некрасиво. Проблема решается очень просто: всю ширину меню (100%) делим на количество разделов, которое равно количеству строк в таблице sections. Для этого воспользуемся оператором ^eval() и тем, что можно использовать объекты класса table в математических выражениях. При этом их числовое значение равно числу записей в таблице. Обратите внимание также на то, что мы пользуемся целочисленным делением, используя обратный слеш вместо прямого.

На ^eval() остановимся чуть подробнее. Он позволяет получить результат математического выражения без введения дополнительных переменных, иными словами, хотим что-то посчитать - пишем:

^eval(выражение)[формат]

Использование [формат] дает возможность вывода результата выражения в том виде, который нужен. Форматная строка [%d] отбрасывает дробную часть, [%.2f] дает два знака после запятой, а [%04d] отводит 4 знака под целую часть, дополняя недостающие символы нулями слева. Форматированный вывод нужен, когда необходимо представить число в определенном виде (скажем, 12.44 $ смотрится куда лучше 12.44373434501 $).

Вот, собственно, и все, что касается меню. Теперь оно функционально и готово к использованию.

Наш первый кирпичик для будущего сайта готов. Теперь займемся структурой страниц. Давайте разобьем их на следующие блоки: header - верхняя часть страницы, body - основной информационный блок, включающий также наше меню и footer - нижняя часть страницы. Многие сайты имеют похожую структуру.

Footer будет для всех страниц одинаковым, header - для всех страниц одинаковый по стилю, но с разными текстовыми строками - заголовками страницы, а body будет разный у всех страниц, сохраняя только общую структуру (предположим, два вертикальных информационных блока, соотносящихся по ширине как 3:7). К body отнесем и наше меню.

Каждая из страниц будет иметь следующую структуру:

header
navigation

body_additional
(30%)

body_main
(70%)
footer
Также, как в случае с меню, опишем каждый из этих блоков методом (функцией) на Parser. Давайте подробно разберемся с каждым блоком.

С footer все очень просто - в auto.p добавляем код:

@footer[]





$now[^date::now[]]

Powered by Parser3
1997-$now.year





Никаких новых идей здесь нет, разве что мы впервые использовали класс date с конструктором now для получения текущей даты, а затем из объекта класса date взяли поле year (год). Если это кажется вам непонятным, обязательно вернитесь к первому уроку, где рассказано о работе с объектами на примере класса table. Все идентично, только теперь мы имеем дело с объектом другого класса.

Немного сложнее с модулем header. С одной стороны, нам нужно формировать уникальный заголовок-приветствие для каждой страницы. В то же время он будет одинаковым с точки зрения внешнего вида, различие только в тексте, который будет выводиться. Как же быть? Мы предлагаем сделать следующее: определить в нашем auto.p новую функцию header, внутри которой будет вызываться другая функция - greeting. А функция greeting, в свою очередь, будет определяться на самих страницах сайта и содержать только то, чем отличаются заголовки страниц (в нашем случае строку-приветствие).

Дополняем auto.p следующим кодом:

@header[]


Тестовый сайт Parser3







^greeting[]



Теперь внимание, кульминация. Parser позволяет сделать очень интересный финт: определить один раз общую структуру страниц в файле auto.p, создать каркас, а затем, используя функции, подобные greeting, в тексте самих страниц, получать разные по содержанию страницы одинаковой структуры. Как это работает?

В самом начале файла auto.p мы определим функцию @main[], которая всегда, причем автоматически, исполняется первой. В нее включим вызовы функций, формирующих части страниц.

В начале auto.p пишем:

@main[]
^header[]
^body[]
^footer[]

А для получения уникального заголовка страниц в каждой из них определим функцию greeting, которая вызывается из header:

для главной страницы:
@greeting[]
Добро пожаловать!

для гостевой книги:
@greeting[]
Оставьте свой след…

и т.д.

Теперь при загрузке, например, главной страницы произойдет следующее:

1. Из файла auto.p автоматически начнет выполняться main.
2. Первой вызывается функция header, из которой вызывается функция greeting.
3. Поскольку функция greeting определена в коде самой страницы, будет выполнена именно она, вне зависимости от того, определяется она в auto.p или нет (происходит переопределение функции).
4. Затем выполняются функции body и footer из main.

В результате мы получаем страницу, у которой будут все необходимые элементы, а в верхней части дополнительно появится наше уникальное приветствие. Переопределяемые функции носят название виртуальных. Мы из файла auto.p вызываем функцию, которая может быть переопределена на любой из страниц и для каждой из них выполнит свой код. При этом общая структура страниц будет абсолютно одинаковой, и сохранится стилистическое и логическое единство.

Осталось описать только основной блок - body. Как мы договорились, он будет состоять из двух частей, каждую из которых будем создавать своей функцией, например, body_main и body_additional, а поскольку навигационное меню, по логике, относится к основной части страниц, вызовем navigation также из body. Снова воспользуемся механизмом виртуальных функций. Редактируем auto.p - дополняем:

@body[]
^navigation[]






^body_additional[]

^body_main[]



Определение функций body_main и body_additional, также как и в случае с greeting вставим в страницы:

@body_additional[]
Главная страница сайта

@body_main[]
Основное содержание

Этот текст приводится как образец для index.html. Отлично! Структура окончательно сформирована. Мы описали все необходимые модули в файле auto.p, сформировали общую структуру и теперь можем запросто генерировать страницы. Больше не нужно помногу писать одни и те же куски HTML кода. Привычные HTML-страницы трансформируются примерно в следующее (примерное содержание index.html файла для главной страницы):

@greeting[]
Добро пожаловать!

@body_additional[]
Главная страница сайта

@body_main[]
Основное содержание

Просто и понятно, не правда ли? Все разложено по полочкам и легко доступно. При этом после обработки подобного кода Parser создаст HTML-код страницы, у которой будет уникальный заголовок, меню, основной информационный блок заданной структуры и footer, одинаковый для каждой страницы. Фактически, мы уже создали готовый сайт, который осталось только наполнить информацией. Это готовое решение для изящного сайта-визитки, который можно создать прямо на глазах. Естественно, это не единственное решение, но такой подход дает отличную структуризацию нашего сайта. Некоторые умственные усилия при разработке структуры с лихвой окупятся легкостью последующей поддержки и модернизации. Каркас хранится в auto.p, а все, что относится непосредственно к странице, - в ней самой.

Дальше открываются безграничные просторы для фантазии. Допустим, вам понадобилось поменять внешний вид заголовка страниц на сайте. Мы открываем auto.p, редактируем один единственный раз функцию @header[] и на каждой из страниц получаем новый заголовок, по стилю идентичный всем остальным. Для обычного HTML нам пришлось бы вручную переписывать код для каждой страницы. Та же самая ситуация и с остальными модулями. Если возникло желание или необходимость изменить общую структуру страниц, например, добавить какой-то блок, достаточно определить его новой функцией и дополнить функцию main в auto.p ее вызовом.

Подобная организация страниц сайта дополняет проект еще одним мощным средством. Предположим, на одной из страниц нам понадобилось получить footer, отличный от других страниц (напомним, изначально мы предполагали, что footer везде одинаковый). Единственное, что нужно сделать, это переопределить функцию footer на нужной странице. Например, такое наполнение /contacts/index.html:

@greeting[]
Наша контактная информация

@body_additional[]
Главная страница тестового сайта

@body_main[]
Основное содержание

@footer[]
Здесь у нас контакты

изменит привычный footer на обозначенный выше, т.е. если Parser находит в тексте страницы код для функции, вызываемой из auto.p, он выполнит именно его, даже если функция определена в самом auto.p. Если же функция не переопределена на странице, то будет использован код из auto.p.

В заключение немного теории для любознательных. Мы будем давать подобную информацию для тех, кто хочет глубже понимать логику работы Parser.

Помните, мы использовали в нашем коде конструкцию $request:uri? Она отличается по синтаксису от всего того, с чем мы имели дело раньше. Что же это такое? Внешне похоже на $объект.свойство (урок 1) - значение полей объекта, только вместо точки использовано двоеточие. На самом деле, это тоже значение поля, только не объекта, а самого класса request. В Parser не предусматриваются конструкторы для создания объектов этого класса. Поля подобных классов формируются самим Parser, а мы можем сразу напрямую обращаться к ним. Техническим языком это называется статическая переменная (поле) uri класса request. Она хранит в себе URI текущей страницы. Также, наряду со статическими переменными, существуют статические методы, с которыми мы столкнемся уже в следующем уроке. При этом можно сразу же вызывать их также без создания каких-либо объектов с помощью конструкторов. Запомните, что в синтаксисе статических полей и методов всегда присутствует двоеточие. Если встречается конструкция вида $класс:поле - мы получаем значение поля самого класса, а запись ^класс:метод является вызовом статического метода класса. Например, для работы с математическими функциями в Parser существует класс math. В нем используются только статические методы и переменные:

$math:PI - возвращает число p. Это статическая переменная класса math.

^math:random(100) - возвращает псевдослучайное число из диапазона от 0 до 100. Это статический метод класса math.

Отличие от записи методов и полей объектов состоит только в двоеточии.

Давайте подведем итоги второго урока.

Первый шаг - раздел новостей

На двух предыдущих уроках мы описали общую структуру нашего сайта. Сейчас это всего лишь каркас. Пора приступать к его информационному наполнению. Практически на каждом сайте присутствует раздел новостей, и наш не должен стать исключением. Также как и с разделами сайта, мы предлагаем начать работу над разделом новостей с создания меню к этому разделу. В качестве средства доступа к новостям будем использовать привычный глазу календарь на текущий месяц.

Создать календарь средствами одного HTML - задача достаточно нетривиальная, к тому же код получится очень громоздким. Сейчас Вы увидите, как легко это сделать на Parser. Приступаем.

Все файлы, относящиеся к разделу новостей, будем размещать в разделе /news/, что было указано нами в файле sections.cfg. Для начала создадим там (!) файл auto.p.Удивлены? Да, файлы auto.p можно создавать в любом каталоге сайта. Однако при этом надо иметь в виду, что функции, описанные в auto.p разделов, будут явно доступны только внутри этих разделов. Согласитесь, ни к чему загромождать корневой auto.p функциями, которые нужны для одного раздела. Логичнее вынести их в отдельный файл, относящийся именно к этому разделу.

Еще одно замечание: если в auto.p раздела переопределить функцию, ранее описанную в корневом auto.p, то будет исполняться именно эта, переопределенная функция. Сработает механизм виртуальных функций, описанный в предыдущем уроке.

Итак, в auto.p раздела news пишем такой код:

@calendar[]
$calendar_locale[
$.month_names[
$.1[Январь]
$.2[Февраль]
$.3[Март]
$.4[Апрель]
$.5[Май]
$.6[Июнь]
$.7[Июль]
$.8[Август]
$.9[Сентябрь]
$.10[Октябрь]
$.11[Ноябрь]
$.12[Декабрь]
]
$.day_names[
$.0[пн]
$.1[вт]
$.2[ср]
$.3[чт]
$.4[пт]
$.5[сб]
$.6[вс]
]
$.day_colors[
$.0[#000000]
$.1[#000000]
$.2[#000000]
$.3[#000000]
$.4[#000000]
$.5[#800000]
$.6[#800000]
]
]
$now[^
date::now[]]
$days[^
date:calendar[rus]($now.year;$now.month)]






^for[week_day](0;6){

}

^days.menu{

^for[week_day](0;6){
^if($days.$week_day){
^if($days.$week_day
==$now.day){

}{

}
}{

}
}

}

$calendar_locale.month_names.[$now.month]


$calendar_locale.day_names.$week_day



$days.$week_day



$days.$week_day

 



Мы определили функцию calendar, которая создает HTML-код календаря. Получился довольно громоздкий код, но ведь и задачи, которые мы ставим перед собой, тоже усложнились. Не волнуйтесь, сейчас во всем разберемся.

Самая объемная часть кода, начинающаяся с определения $calendar_locale, оказалась незнакомой. Посмотрите на эту структуру. Похоже, в ней мы определяем какие-то данные для календаря, напоминающие таблицу. То, что определено как $calendar_locale, в терминологии Parser называется «хеш», или ассоциативный массив. Зачем он нужен можно сказать, просто бегло просмотрев код примера: здесь мы сопоставляем русское написание месяца его номеру в году (3 - март), название дня недели его номеру, а также связываем шестнадцатиричное значение цвета с некоторым числом. Теперь идея хешей должна проясниться: они нужны для сопоставления (ассоциативной связи) имени с объектом. В нашем случае мы ассоциируем порядковые номера месяцев и дней с их названиями (строками). Parser использует объектную модель, поэтому строка тоже является объектом. Нам несложно получить порядковый номер текущего месяца, но намного нагляднее будет вывести в календаре «Ноябрь» вместо «11» или «пн» вместо «1». Для этого мы и создаем ассоциативный массив.

В общем виде порядок объявления переменных-хешей такой:

$имя[
$.ключ[значение]
]

Эта конструкция позволяет обратиться к переменной по имени с ключом $имя.ключ и получить сопоставленное значение. Обратите внимание, что в нашем случае мы имеем хеш, полями которого являются три других хеша.

После определения хеша мы видим уже знакомую переменную now (текущая дата), а вот дальше идет незнакомая конструкция:

$days[^date:calendar[rus]($date.year;$date.month)]

По логике работы она напоминает конструктор, поскольку в переменную days помещается таблица с календарем на текущий месяц текущего года. Тем не менее, привычного :: здесь не наблюдается. Это один из статических методов класса date. Статические методы наряду с уже знакомыми конструкторами могут возвращать объекты, поэтому в данном случае необходимо присвоить созданный объект переменной. Про статические переменные и методы уже было немного сказано в конце предыдущего урока. Своим появлением они обязаны тому факту, что некоторые объекты или их свойства (поля) существуют в единственном экземпляре, как, например, календарь на заданный месяц или URI страницы. Поэтому подобные объекты и поля выделены в отдельную группу, и к ним можно обращаться напрямую, без использования конструкторов. В случае если мы обращаемся к статическому полю, мы получаем значение поля самого класса. В качестве примера можно привести класс math, предназначенный для работы с математическими функциями. Поскольку существует только одно число p, то для того, чтобы получить его значение, используется статическое поле $math:PI - это значение поля самого класса math.

В результате исполнения этого кода в переменной days будет содержаться такая таблица:
Таб.1 (для ноября 2001 года)

0

1

2

3

4

5

6



01

02

03

04

05

06

07

08

09

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

Это таблица, содержащая порядковый номер дней недели и календарь на 11.2001.

С ней мы и будем дальше работать. Нельзя сразу же выводить содержимое переменной $days, просто обратившись к ней по имени. Если мы обратимся к таблице просто как к переменной, будет непонятно, что мы хотим получить - строку, всю таблицу целиком или значение только из одного столбца. Также явно требуется доработка содержимого полученной таблицы. Но ведь не зря же мы создавали наш хеш с названиями дней недели и месяцев. Поэтому далее по коду средствами HTML создается таблица, в первой строке которой мы выводим название текущего месяца, пользуясь данными из хеша, связанными с номером месяца в году:

$calendar_locale.month_names.[$now.month]

Что здесь что? Мы выводим значение поля month_names хеша calendar_locale с порядковым номером текущего месяца, полученным как $now.month. Результатом выполнения этой конструкции будет название месяца на русском (или любом другом) языке, которое было определено в хеше.

В следующей строке календаря выведем названия дней недели, пользуясь данными хеша. Давайте чуть подробнее определимся с задачей. Нам надо последовательно перебрать номера дней недели (от 0 до 6) и поставить в соответствие номеру дня его название из поля day_names хеша calendar_locale. Для этой цели удобнее всего воспользоваться циклом: последовательностью действий, выполняющейся заданное количество раз. В данном случае мы используем цикл for. Его синтаксис такой:

^for[счетчик](диапазон значений, например 0;6){последовательность действий}

Одно из достоинств циклов - возможность использования значения счетчика внутри цикла, обращаясь к нему как к переменной. Этим мы и воспользуемся:

^for[week_day](0;6){


$calendar_locale.day_names.$week_day


}

Все просто и понятно, если знать, что такое цикл: последовательно меняя значение week_day от 0 до 6 (здесь week_day является счетчиком цикла), мы получаем семь значений:

$calendar_locale.day_colors.$week_day - для цвета шрифта
$calendar_locale.day_names.$week_day - для названия дня недели.

Идея получения данных та же, что и для получения названия месяца, только используются другие ключи хеша.

Возможно, возник вопрос: зачем в хеше ключ day_colors? Ответ прост - все должно быть красиво! Если есть возможность максимально приблизить наш календарь к реальному, то так и сделаем - перекрасим выходные дни в красный цвет.

Далее по тексту следует большой красивый блок. Чтобы в нем разобраться, определимся с задачами. Итак, нам нужно:

1. Последовательно перебрать строки таблицы days (Таб.1).
2. В каждой строке таблицы days последовательно перебрать и вывести значения столбцов (числа месяца).
3. Корректно вывести пустые столбцы (то есть проверить первую и последнюю недели месяца на полноту).
4. Как-то выделить текущее число, например другим цветом и жирным шрифтом.

Приступаем. Первый пункт решается с помощью знакомого метода menu класса table:

^days.menu{…}

Перебор столбцов логичнее всего сделать циклом for, с которым мы только что познакомились:

^for[week_day](0;6){…}

Для проверки столбцов на пустоту для вывода столбцов без чисел используем оператор if. Вообще, любые проверки условий всегда можно реализовать с помощью if:

^if($days.$week_day){

}{
 
}

Обратите внимание, что в условии if мы ни с чем не сравниваем $days.$week_day. Так осуществляется проверка на неравенство нулю.

Parser это условие понимает так:

«Если существует $days.$week_day, то {…}, если нет, то вывести пустую ячейку таблицы серого цвета»

Основная часть работы выполнена. Осталось только выделить текущее число. Решается это использованием еще одного if, где условием задается сравнение текущего значения таблицы days с текущим числом ($days.$week_day==$now.day):

^if($days.$week_day==$now.day){


$days.$week_day


}{


$days.$week_day


}

Обратите внимание на то, что здесь мы проверяем на равенство два числа, поэтому используем оператор == вместо eq, используемый для проверки равенства строк.

Еще раз посмотрим на общую структуру формирования календаря:

#перебираем строки таблицы с календарем
^days.
menu{

#перебираем столбцы таблицы с календарем
^
for[week_day](0;6){
^if($days.week_day){
^if($month.$week_day==$date.day){
число на другом фоне жирным шрифтом
}{
число
}
}{
пустой серый столбец
}
}
}


Эту конструкцию простой не назовешь. Здесь используются вложенные друг в друга конструкции. Однако она позволяет понять возможность комбинирования различных средств Parser для решения конкретной задачи. Есть более элегантное решение - вынести проверку текущей даты и ее раскрашивание нужным цветом в отдельную функцию, которую и вызывать внутри цикла. Похожее решение мы использовали во втором уроке. Это позволит разгрузить блок и сделать его более читабельным. Но поскольку в данном примере мы хотели показать вам возможность комбинирования нескольких логических структур, то оставляем эту возможность вам в качестве задания.

Если хотите убедиться в работоспособности этого модуля, создайте в разделе /news/ файл test.html и в нем наберите одну единственную строчку ^calendar[]. Теперь откройте этот файл из браузера и полюбуйтесь результатом своих трудов.

Подведем итоги третьего урока.

Шаг второй - переходим к работе с БД

Самое главное - не пугайтесь названия урока, даже если вы никогда не работали с базами данных. Без них просто невозможно построить гибкий, легко настраиваемый сайт. Отказ от использования БД не дает никаких преимуществ разработчику, а наоборот, здорово уменьшает возможности по созданию сайта и быстрому динамическому изменению содержимого. Построение серьезного ресурса без БД - это как рыбалка без удилища: вроде бы и можно кого-то поймать, однако делать это крайне неудобно. Иными словами, если вы пока не умеете работать с БД - обязательно научитесь и активно используйте их в своих проектах. На этом закончим агитацию, будем считать, что мы вас убедили в необходимости использования БД.

Работать с БД на Parser очень удобно. В Parser встроена мощная система взаимодействия с различными СУБД. В настоящее время Parser может работать с MySQL, Oracle, PgSQL, а также с любой СУБД через драйверы ODBC (в т.ч. MS SQL, MS Access). Поскольку исходные коды Parser3 являются открытыми, возможно добавление поддержки любых других знакомых вам СУБД после создания соответствующего драйвера. При этом работа с ними не требует практически никаких дополнительных знаний собственно Parser. Все, что нужно - это подключится к выбранной СУБД и работать, используя SQL в объеме и формате, поддерживаемом СУБД. При передаче SQL-запросов Parser может только заменить апострофы соответствующей конструкцией в зависимости от СУБД, для «защиты от дурака», а все остальное передается, как есть.

Существует еще специальная конструкция для записи больших строковых литералов. Oracle, PgSQL и, возможно, какие-то серверы, драйверы к которым будут написаны в будущем, не умеют работать с большими строковыми литералами. Если передаваемая, например, из формы, строка будет содержать больше 2000 [Oracle 7.x] или 4000 [Oracle 8.x] букв, сервер выдаст ошибку «слишком длинный литерал». Если пытаться хитрить, комбинируя «2000букв» + «2000букв», то также будет выдана ошибка «слишком длинная сумма». Для хранения таких конструкций используется тип данных CLOB[Oracle] и OID[PgSQL], а для того, чтобы SQL команды были максимально просты, при записи таких строк необходимо лишь добавить управляющий комментарий, который драйвер соответствующего SQL-сервера соответствующим образом обработает:

insertinto news text values (/**text**/'$form:text')

Слово text в записи /**text**/ - это имя колонки, в которую предназначен следующий за этой конструкцией строковый литерал. Пробелы здесь недопустимы!

Со всеми возможностями Parser по работе с различными СУБД в рамках данного урока мы знакомиться, конечно же, не будем. Остановимся на MySQL. Почему именно на ней? Прежде всего потому, что она очень распространена, и многие веб-проекты используют именно ее. Кроме того, практически все компании, занимающиеся сетевым хостингом, предоставляют клиентам возможность работы с этой СУБД. Ну и, несомненно, немаловажный фактор - она бесплатна, доступна и легка в освоении.

Давайте определимся, что будем хранить в базе данных. Очевидный ответ : будем хранить новости. Причем таблица СУБД с новостями должна содержать такие поля: уникальный номер новости в базе, который будет формироваться автоматически СУБД, дата внесения новости в базу, по которой мы будем проводить выборку новостей за конкретное число, заголовок новости и собственно ее текст. Просто, без тонкостей и премудростей, однако это эффективно работает.

Есть еще один вопрос, с которым нужно определиться: каким образом новости будут попадать в базу? Можно их заносить и из командной строки СУБД, но это не удобно. В случае если вы предполагаете строить сайт для intranet, есть вариант использовать в качестве СУБД или средства доступа к БД широко распространенную MS Access. Привычный GUI и copy+paste обеспечат вам любовь многих коллег по работе на долгие годы. Для маленьких баз данных это решение может оказаться оптимальным. Мы же предлагаем решение, ориентированное на Internet - создание на сайте раздела администрирования с формой для ввода новостей прямо из браузера.

Постановка задачи закончена, переходим к ее практическому решению. Для дальнейшей работы вам потребуется установленная СУБД MySQL, без которой рассматриваемый здесь пример просто не будет работать.

Прежде всего, средствами MySQL создаем новую базу данных с именем p3test, содержащую одну единственную таблицу news с полями id, date, header, body:

id
int not null auto_increment primary key
date
date
header
varchar(255)
body
text

Теперь создадим раздел администрирования, который даст возможность заполнить созданную базу данных новостями. Для этого в корневом каталоге сайта создаем каталог admin, а в ней index.html, в который пишем следующее:

@greeting[]
Администрирование новостей

@body_additional[]
Добавление новостей

@body_main[]
$now[^
date::now[]]




Date:
     
Header:


Body:





   





#начало обработки
^if(def $form:date && def $form:header && def $form:body){
^
connect[$connect_string]{
^
void:sql{insert into news
(date,
header, body)
values
('$form:date',
'$form:header', '$form:body')
}

…сообщение добавлено
}
}{
…для добавления новости необходимо заполнить все поля формы
}


Также требуется в корневом файле auto.p перед методом main добавить метод auto. Этот метод используется для инициализации глобальных переменных, т.е. переменных, которые будут доступны на всех страницах сайта. В нем мы зададим строку подключения к базе данных, о которой чуть позже.

@auto[]
$connect_string[
mysql://root@localhost/p3test]

Как видите, структура этой страницы полностью соответствует придуманной нами структуре страниц сайта. Все элементы, как то: приветствие, две части body, footer и header присутствуют. Кстати, вы помните, откуда на этой странице появятся header и footer? Правильно, из функции main корневого auto.p).

Незнакомые конструкции только в основной части. Давайте с ней разберемся. В начале обычная HTML форма, с подстановкой текущей даты в поле date как значения по умолчанию. Сделано это исключительно для удобства пользователей.

Легкое недоумение может вызвать запись:
${now.year}-${now.month}-${now.day}

Фигурные скобки здесь используются для того, чтобы получить строку вида «2001-11-06» (в таком формате мы собираемся хранить дату новости в БД). Если скобки не ставить, то Parser выдаст ошибку при обработке этого кода, поскольку не сможет понять, что нужно делать. Для него «-» будет частью имени. Запомните, если вам нужно четко отделить имя переменной от следующей за ним буквы, скажем «-», как в нашем случае, нужно записать:

${имя_переменной}-

И в результате вы получите:

значение_переменной-

Обязательно прочитайте страницу, посвященную правилам составления имен.

Лучшим решением этой проблемы было бы использовать в этом месте конструкцию ^date.sql-string[]. Попробуйте самостоятельно доработать этот пример, пользуясь справочником. Если не получится - не расстраивайтесь, на следующем уроке мы покажем, как это сделать.

Продолжим. Если вам уже доводилось работать с формами, то вы знаете, что формы передают введенные в них значения на дальнейшую обработку каким-либо скриптам. Здесь обработчиком данных формы будет сама страница, содержащая эту форму. Никаких дополнительных скриптов нам не понадобится.

После закрывающего тега начинается блок обработки. Вначале с помощью if мы проверяем поля формы на пустоту. Этого можно опять же не делать, но мы хотим создать нечто большее, чем учебный экспонат без практического применения. Для того чтобы осуществить проверку, необходимо получить значения полей этой формы. В Parser это реализуется через статические переменные (поля). Мы просто обращаемся к полям формы, как к статическим полям:

$form:поле_формы

Полученные таким образом значения полей мы и будем проверять на пустоту с помощью оператора def и логического «И»(&&). Мы уже проверяли объект на существование в третьем уроке, но там был опущен оператор def, поскольку проверяли на пустоту таблицу. Как вы помните, таблица в выражении имеет числовое значение, равное числу строк в ней, поэтому любая непустая таблица считается определенной. Здесь же необходимо использовать def, как и в случае проверки на def других объектов. Если в поле ничего не было введено, то значение $form:поле_формы будет считаться неопределенным (undefined). После того, как все значения полей заполнены, необходимо поместить их в базу данных. Для этого нужно сначала подключиться к базе данных, а затем выполнить запрос SQL для вставки данных в таблицу. Посмотрите, как мы это делаем:

^connect[$connect_string]{
^void:sql{insert into news
(date, header, body)
values
('$form:date', '$form:header', '$form:body')
}

…cообщение добавлено
}

Удобство Parser при работе с базами данных состоит в том, что он, за исключением редких случаев, не требует изучать какие-либо дополнительные операторы, кроме тех, которые предусмотрены в самой СУБД. Сессия работы с базой данных находится внутри оператора connect, общий синтаксис которого:

^connect[протокол://строка соединения]{методы, передающие запросы SQL}

Для MySQL это запишется так:

^connect[mysql://пользователь:пароль@хост/база_данных]{…}

В фигурных скобках помещаются методы, выполняющие SQL-запросы. При этом любой запрос может возвратить или не возвратить результат (например, в нашем случае нужно просто добавить запись в таблицу БД, не возвращая результат), поэтому Parser предусматривает различные конструкции для создания этих двух типов SQL-запросов. В нашем случае запрос записывается как:

^void:sql{insert into news
(date, header, body)
values
('
$form:date', '$form:header', '$form:body')
}

Кстати, это статический метод класса void, помните про двоеточие?

То, что здесь не выделено цветом, является командами SQL. Ничего сложного здесь нет. Если вы знакомы с SQL, то больше ничего и не потребуется, а если почему-то пока не знакомы, мы вновь рекомендуем его изучить. Вам это многократно пригодится в дальнейшем. Время, потраченное на это изучение, не пропадет даром.

Оцените все изящество этого варианта взаимодействия с базой данных - Parser обеспечивает прозрачный доступ к СУБД и, за редким исключением, не требует каких-либо дополнительных знаний. При этом, как вы видите, мы можем помещать в запросы SQL еще и данные из нашей формы, пользуясь конструкциями Parser. Возможности этого симбиоза просто безграничны. СУБД решает все задачи, связанные с обработкой данных (она ведь именно для этого и предназначена и очень неплохо с этим справляется), а нам остается только воспользоваться результатами ее работы. Все аналогично и с другими СУБД, с которыми вы можете столкнуться в своей работе.

Теперь у нас есть форма, позволяющая помещать записи в нашу БД. Занесите в нее несколько записей. А теперь давайте их оттуда извлекать, но перед этим неплохо бы немного доработать функцию calendar, созданную на предыдущем уроке. Нужно, чтобы в календаре ставились ссылки на дни месяца, а выбранный день передавался как поле формы. Тогда по числам-ссылкам в календаре пользователь будет попадать в архив новостей за выбранный день. Модернизация эта неcложная, просто добавим немного HTML в auto.p раздела news: $days.$week_day в коде if обнесем ссылками таким образом:

$days.$week_day

В результате мы получаем возможность использовать наш календарь в качестве меню доступа к новостям за определенный день.

Теперь займемся /news/index.html. В него заносим такой код:

@greeting[]
Страница новостей, заходите чаще!

@body_additional[]
Архив новостей за текущий месяц:


^calendar[]

@body_main[]

НОВОСТИ


$day(^if(def $form:day){
$
form:day
}{
$now.
day
}
)
^
connect[$connect_string]{
$news[^
table::sql{select
date, header, body
from
news
where
date='${now.
year}-${now.month}-$day'
}]
^if($news){
^news.
menu{
$news.date - $news.header

^
untaint{$news.body}

}[

]
}{
За указанный период новостей не найдено.
}
}

Структура обычная. В дополнительной части body помещаем меню-календарь вызовом ^calendar[] (напомним, что эта функция определена в auto.p раздела news). Основа информационной части страницы - выборка из базы данных новостей за тот день, по которому щелкнул пользователь (условие where в SQL-запросе). Это второй вариант SQL-запроса, при котором результат возвращается. Обратите внимание, здесь результатом запроса будет таблица, с которой в дальнейшем мы будем работать. Поэтому необходимо создать объект класса table.

Познакомимся с еще одним конструктором класса table - конструктором на базе SQL-запроса. Его логика абсолютно аналогична работе конструктора ^table::load[], только источником данных для таблицы является не текстовый файл, как в случае с пунктами меню, а результат работы SQL-запроса - выборка из базы данных:

$переменная[^table::sql{код SQL запроса}]

Воспользоваться этим конструктором вы можете только внутри оператора ^connect[], то есть когда имеется установленное соединение с базой данных, поскольку обработкой SQL-запросов занимается сама СУБД. Результатом будет именованная таблица, имена столбцов которой совпадают с заголовками, возвращаемыми SQL-сервером в ответ на запрос.

Небольшое отступление. При создании SQL-запросов следует избегать использования конструкций вида select * from … поскольку для постороннего человека, не знающего структуру таблицы, к которой происходит обращение, невозможно понять, какие данные вернутся из БД. Подобные конструкции можно использовать только для отладочных целей, а в окончательном коде лучше всегда явно указывать названия полей таблиц, из которых делается выборка данных.

Остальная часть кода уже не должна вызывать вопросов: if обрабатывает ситуацию, когда поле day (выбранный пользователем день на календаре, который передается из функции calendar) не определено, то есть человек пришел из другого раздела сайта через меню навигации. Если поле формы day определено (def), то используется день, переданный посетителем, в противном случае используем текущее число. Далее соединяемся с БД, также как мы это делали, когда добавляли новости, создаем таблицу $news, в которую заносим новости за запрошенный день (результат SQL-запроса), после чего с помощью метода menu последовательно перебираем строки таблицы news и выводим новости, обращаясь к ее полям. Все понятно и знакомо, кроме одного вспомогательного оператора, который служит для специфического вывода текста новости:

^untaint{$news.body}

Отвлекитесь немного и внимательно прочитайте раздел справочника, посвященный операторам taint и untaint, где подробно описана логика их работы. Это очень важные операторы и вы наверняка столкнетесь с необходимостью их использования. К тому же большой объем работы по обработке данных Parser делает самостоятельно, она не видна на первый взгляд, но понимать логику действий необходимо.

Прочитали? Теперь продолжим. Зачем он нужен здесь? У нас есть страница для администрирования новостей, и мы хотим разрешить использование тегов HTML в записях. По умолчанию это запрещено, чтобы посторонний человек не мог внести Java-скрипт, например, перенаправляющий пользователя на другой сайт. Как это сделать? Да очень просто: достаточно выборку записей из таблицы преобразовать с помощью оператора untaint:

^untaint{текст новости}

В нашем случае используется значение по умолчанию [as-is], которое означает, что данные будут выведены так, как они есть в базе. Мы можем позволить себе поступить так, поскольку изначально не предполагается доступ обычных пользователей к разделу администрирования, через который добавляются новости.

Теперь можно немного расслабиться - новостной блок нашего сайта завершен. Мы можем добавлять новости и получать их выборку за указанный пользователем день. На этом четвертый урок будем считать оконченным, хотя есть некоторые детали, которые можно доработать, а именно: научить календарь не ставить ссылки на дни после текущего, выводить в заголовке информационной части дату, за которую показываются новости, да и просто реализовать возможность доступа к новостям не только за текущий месяц, но и за предыдущие. Однако это уже задание вам. Знаний, полученных на предыдущих уроках вполне достаточно, чтобы доработать этот пример под свои требования и желания. Творите!

Подведем итоги четвертого урока.

Пользовательские классы Parser

Во всех предыдущих уроках мы оперировали классами и объектами, предопределенными в Parser. Например, есть уже хорошо знакомый нам класс table, у него существуют свои методы, которые мы широко использовали. Полный список всех методов этого класса можно посмотреть в справочнике. Однако ограничение разработчиков рамками только базовых классов в какой-то момент может стать сдерживающим фактором. «Неспособность не есть благодетель, а есть бессилие…», поэтому для удовлетворения всех потребностей пользователей необходимо иметь возможность создавать собственные (пользовательские) классы объектов со своими методами. На этом уроке мы и создадим средствами Parser новый класс объектов со своими собственными методами.

Объектом, в принципе, может быть все что угодно: форум, гостевая книга, различные разделы и даже целый сайт. Здесь мы подошли к очередному уровню структуризации - на уровне объектов, а не методов. Как мы поступали раньше? Мы выделяли отдельные куски кода в методы и вызывали их, когда они были необходимы. Но в качестве отдельных блоков сайта было бы намного удобнее использовать собственные объекты: для получения форума создаем объект класса «форум», после чего используем его методы, например «удалить сообщение», «показать все сообщения» и поля, например, «количество сообщений». При этом обеспечивается модульный подход на качественно ином уровне, чем простое использование функций. В единую сущность собираются код и данные (методы и поля). Разрозненные ранее методы и переменные объединяются воедино и используются применительно к конкретному объекту - «форуму». В терминах объектно-ориентированного программирования это называется инкапсуляцией. Кроме того, один раз создав класс форум, его объекты можно использовать в различных проектах, абсолютно ничего не меняя.

Работу с пользовательским классом мы покажем на примере гостевой книги, а для начала еще раз напомним порядок работы с объектами в Parser. Сначала необходимо создать объект определенного класса с помощью конструктора, после чего можно вызывать методы объектов этого класса и использовать поля созданного объекта. В случае пользовательского класса ничего не меняется, порядок тот же.

Как всегда начнем с определения того, что нам нужно сделать. Правильная постановка задачи - уже половина успеха. Перед началом создания класса нужно точно определить, что будет делать объект класса, то есть решить, какие у него будут методы. Предположим, что нашими методами будут: показ сообщений гостевой книги, показ формы для добавления записи, а также метод, добавляющий сообщение в гостевую книгу. Хранить сообщения будем в базе данных, так же как и новости.

Если с методами класса все более или менее ясно, то некоторая неясность остается с конструктором класса, что же он будет делать? Опираясь на прошлые уроки, мы помним, что для того, чтобы начать работать с объектом класса, его необходимо создать, или проинициализировать. Давайте с помощью конструктора будем получать таблицу с сообщениями, а затем в методе показа сообщений будем пользоваться данными этой таблицы.

С целями определились, займемся реализацией. Прежде всего, создадим таблицу gbook в базе данных p3test:

id
int not null auto_increment primary key
author
varchar(255)
email
varchar(255)
date
date
body
text

Теперь необходимо познакомиться еще с несколькими понятиями Parser - классом MAIN и наследованием. Как уже говорилось, класс является объединяющим понятием для объектов, их методов и полей. Класс MAIN объединяет в себя методы и переменные, описанные пользователями в файлах auto.p и запрашиваемом документе (например, index.html). Каждый следующий уровень вложенности наследует методы, описанные в auto.p предыдущих уровней каталога. Эти методы, а также методы, описанные в запрашиваемом документе, становятся статическими функциями класса MAIN, а все переменные, созданные в auto.p вверх по каталогам и в коде запрошенной страницы, - статическими полями класса MAIN.

Для пояснения рассмотрим следующую структуру каталогов:

/
|__ auto.p
|__ news/
| |___auto.p
| |___index.html
| |___details/
| |_______ auto.p
| |_______index.html
|__contacts/ |
|_______auto.p
|_______index.html



При загрузке страницы index.html из каталога /news/details/ класс MAIN будет динамически «собран» из методов, описанных в корневом файле auto.p, а также в файлах auto.p разделов /news/ и /news/details/. Методы, описанные в auto.p раздела /contacts/, будут недоступны для страниц из раздела /news/details/.

Как «собирается» класс MAIN теперь понятно, но, прежде чем приступить к созданию собственного класса, необходимо также выяснить, как из пользовательского класса вызывать методы и получать значения переменных класса MAIN. Методы класса MAIN вызываются как статические функции:

^MAIN:метод[],

а переменные являются статическими полями класса MAIN. К ним можно получить доступ так же, как к любым другим статическим полям:

$MAIN:поле

Теперь переходим к практике. В корневой auto.p добавляем еще один метод, с помощью которого можно будет не только соединяться с БД, но и передавать ей произвольный SQL-запрос:

@dbconnect[code]
^
connect[$connect_string]{$code}
# connect_string определяется в методе @auto[]
# $connect_string[
mysql://root@localhost/p3test]

Метод вынесен в корневой auto.p для того, чтобы впоследствии можно было бы легко подключаться к серверу баз данных с любой страницы, поскольку методы из корневого auto.p будут наследоваться всегда. Обратите внимание на то, что здесь используется передача методу параметра. В нашем случае он один - code, с его помощью мы и будем передавать код, выполняющий SQL-запросы. Параметров может быть и несколько, в этом случае они указываются через точку с запятой.

Дальше в каталоге нашего сайта создаем подкаталог, в которой будем хранить файл с нашим классом, например, classes. Далее в этом каталоге создаем файл gbook.p (пользовательские файлы мы предлагаем хранить в файлах с расширением имени .p) и в него заносим следующий код:

@CLASS
gbook

@load[]
^MAIN:dbconnect{
$messages[^
table::sql{select author, email, date, body from gbook}]
}

@show_messages[]
^if($messages){
^messages.
menu{





$messages.author
^if(def $messages.email){
$messages.email
}{
Нет электронного адреса
}
$messages.date





$messages.body

}[

 ^;
]
}{
Гостевая книга пуста.
}

@show_form[]



$date[^date::now[]]

"POST">

Author*

E-mail  

text




   






@test_and_post_message[]
^if(
def $form:post){
^if(def $form:author){
^MAIN:dbconnect{
^void:sql{insert into gbook
(author, email, date, body)
values (
'$form:author',
'$form:email',
'${date.year}-${date.month}-${date.day}',
'$form:text'

)}
}
$response:location[$request:uri]
}{
Поле автор обязательно для заполнения

}
}

Посмотрите на код. В первой строке мы говорим, что в этом файле будем хранить пользовательский класс:

@CLASS

В том случае если в качестве родительского выступает другой пользовательский класс, его необходимо подключить, а также объявить базовым. Получится такая конструкция:

@CLASS
имя класса

@USE
файл родительского класса

@BASE
имя родительского класса

Следующей строкой пишем имя нашего класса gbook. Необходимо помнить, что Parser чувствителен к регистру букв в именах, поэтому классы gbook и Gbook являются разными. При этом имя не обязательно должно совпадать с именем файла, в котором храниться пользовательский класс, более того, может быть набрано русскими буквами.

Дальше определяются методы нашего нового класса. Делается это точно так же, как и определение обычных методов, которые мы создавали на предыдущих уроках.

Первый метод load будет конструктором нашего класса. При этом надо иметь в виду, что задача конструктора - создать объект. Кроме этого, он может объявить переменные и присвоить им значения. Эти переменные станут полями объекта пользовательского класса. В нашем случае мы при помощи конструктора sql класса table создаем нужную таблицу. Обратите внимание, что в методах нового класса мы свободно пользуемся методами системных классов и методом dbconnect класса MAIN:

@load[]
^MAIN:dbconnect{
$messages[^table::sql{select author, email, date, body from gbook}]
}

Как уже говорилось выше, поскольку мы находимся за пределами класса MAIN, для использования методов этого класса перед именем необходимо указать класс, к которому эти методы/поля относятся. Делается это таким образом:

^имя_класса:метод[параметры]
$имя_класса:переменная

В случае, если мы захотим использовать методы/поля другого пользовательского класса, а не класса MAIN, необходимо в начале кода выполнять инструкцию:

@USE
путь к файлу, описывающему класс

Она позволяет использовать модуль, определенный в другом файле. О работе Parser с путями к файлам, рассказано в приложении 1.

Итак, наш новый конструктор будет создавать таблицу с сообщениями, подключаясь к указанной БД. С конструктором разобрались, начинаем описание собственно методов нового класса. Метод show_messages нашего класса выводит на экран сообщения из таблицы gb, созданной в методе load. Строки перебираются при помощи метода menu класса table. Все знакомо, ничего нового нет и в других методах:

show_form - выводит на экран форму для добавления нового сообщения гостевой книги

test_and_post_message - проверяет, нажата ли кнопка post, заполнено ли поле author и, если все условия выполнены, заносит сообщение в базу данных, используя все тот же метод dbconnect, определенный в классе MAIN

На этом создание пользовательского класса, описывающего методы объектов класса gbook, завершено. Его осталось только подключить для использования на нашем сайте. Перед нами стоит задача сообщить Parser, что на некоторой странице мы собираемся использовать свой класс. Для этого в файле index.html каталога gbook в первой строке напишем следующее:

@USE
/classes/gbook.p

Теперь на этой странице можно создать объект класса gbook и использовать затем его методы. Сделаем это в основной информационной части:

@body_main[]
Гостевая книга тестового сайта




$gb[^gbook::load[]]
^gb.show_messages[]
^gb.show_form[]
^gb.test_and_post_message[]

# и конечно же не забываем про остальные части
@greeting[]
Оставьте свой след:

@body_additional[]
Нам пишут…

Здесь мы уже работаем с объектом созданного пользовательского класса, как с любым другим объектом: создаем его при помощи конструктора класса и вызываем методы, определенные в новом классе. Посмотрите, насколько изящным получилось наше решение. Читабельность кода очевидна и, глядя на этот фрагмент, сразу понятно, что он делает. Все, что относится к гостевой книге, находится в отдельном файле, где описано все, что можно с ней сделать. Если нам понадобится новый метод для работы с гостевой книгой, нужно просто дописать его в файл gbook.p. Все очень легко модернизируется, к тому же сразу понятно, где необходимо вносить изменения, если они вдруг понадобились.

В заключение хочется заметить, что изящнее было бы вынести методы вроде dbconnect из класса MAIN в отдельный класс. Это позволило бы не перегружать класс MAIN, улучшилась бы читаемость кода, а также легче стало бы ориентироваться в проекте. Там, где эти нужны методы этого класса, его можно было бы подключать с помощью @USE.

Подведем итоги пятого урока.

Работаем с XML




</b><b>Урок 6. Работаем с XML</b><b>

Представьте, что вам позволено придумывать любые теги
с любыми атрибутами. То есть вы сами можете определять,
что означает тот или иной выдуманный вами тег или атрибут.

Такой код будет содержать данные, …


Класс xdoc
Класс xnode



…но не их форматирование. Подготовкой данных может заняться один человек, а форматированием другой. Им достаточно договориться об используемых тегах и можно приступать к работе… одновременно.

Идея эта не нова, существовали многочисленные библиотеки обработки шаблонов, а многие создавали собственные. Библиотеки были несовместимы между собой, зависели от используемых средств скриптования, порождая разобщенность разработчиков и необходимость тратить силы на изучение очередной библиотеки вместо того, чтобы заняться делом.

Однако прогресс не стоит на месте, и сейчас мы имеем не зависящие от средства скриптования стандарты XML и XSLT, позволяющие нам реализовать то, что мы только что представляли. А также стандарты DOM и XPath, открывающие для нас еще больше возможностей.
Parser полностью поддерживает все эти стандарты.

Сейчас откройте выбранную вами вчера в книжном магазине книгу, описывающую XML и XSLT. Используйте ее как справочник.

Посмотрим, как можно приведенную статью преобразовать из XML в HTML.
Запишем текст из начала статьи в файл article.xml
И создадим файл article.xsl, в котором определим выдуманные нами теги:






</b><b><xsl:value-of select="title" /></b><b>










Ссылки по теме:


















  • Данные и шаблон преобразования готовы. Создаем article.html, в который заносим следующий код:

    # входной xdoc документ
    $sourceDoc[^xdoc::load[article.xml]]

    # преобразование xdoc документа шаблоном article.xsl
    $transformedDoc[^sourceDoc.transform[article.xsl]]

    # выдача результата в HTML виде
    ^transformedDoc.string[
    $.method[html]
    ]

    Первой строкой мы загружаем XML-файл, получая в sourceDoc его DOM-представление.
    Конструкция похожа на загрузку таблицы, помните ^table::load[…]? Только в этот раз мы загружаем не таблицу (получая объект класса table), а XML-документ (получаем объект класса xdoc).

    Второй строкой мы преобразуем входной документ по шаблону article.xsl.
    Из входного документа получаем выходной, применяя XSLT преобразование, описанное в шаблоне.

    Последней строкой мы выдаем пользователю текст выходного документа в HTML формате (параметр method со значением html).
    Здесь можно задать все параметры, допустимые для тега .
    Рекомендуем также задать параметр "без отступов" (параметр indent со значением no: $.indent[no]), чтобы избежать известной проблемы с пустым местом перед .

    Обратившись теперь к этой странице, пользователь получит результат преобразования:

    </b><b>Урок 6. Работаем с XML</b><b>

    Представьте, что вам позволено придумывать любые теги
    с любыми атрибутами. То есть вы сами можете определять,
    что означает тот или иной выдуманный вами тег или атрибут.



    Такой код будет содержать данные, …


    Ссылки по теме:

  • Класс xdoc

  • Класс xnode





  • Как вы заметили, тег никак не был определен, как следствие, информация об авторе статьи не появилась в выходном HTML. Со временем, когда вы решите где и как будете хранить и показывать данные об авторах, достаточно будет дополнить шаблон - исправлять данные статей не потребуется.

    Внимание: если вы не хотите, чтобы пользователи вашего сервера имели доступ к .xml и .xsl файлам, храните эти файлы вне веб-пространства (^xdoc::create[/../directory_outside_of_web_space/article.xml]), или запретите к ним доступ средствами веб-сервера (пример запрета доступа к .p файлам здесь: «Установка Parser на веб-сервер Apache. CGI скрипт»).

    Подведем итоги шестого урока.

    А что это вообще такое?

    Abs, sign. Операции со знаком

    Выполняют операции со знаком числа.

    ^math:abs(число)
    - возвращает абсолютную величину числа (модуль)

    ^math:sign(число)
    - возвращает 1, если число положительное, -1, если число отрицательное и 0, если число равно 0




    Число Пи

    $math:PI - число p




    Compact. Сборка мусора [3.1]

    ^memory:compact[]

    Собирает так называемый «мусор» в памяти, освобождая ее для повторного использования вашим кодом.
    Мусором считается память, более не используемая вашим кодом, т.е. та, на которую в вашем коде нет ссылок.

    Например,
    $table[^table::sql{запрос}]
    $table[]
    # освободит память, занимаемую результатом выполнения SQL-запроса
    ^memory:compact[]

    Parser не собирает мусор автоматически, полагаясь в данном вопросе на кодера: поставьте вызов compact в той точке (точках), где ожидаете наибольшей выгоды, например, перед XSL преобразованием.

    $status:memory поможет вам в отладке и поиске мест, наиболее выгодных для сборки мусора.

    Важно: необходимо как можно более интенсивно использовать локальные переменные, и обнулить глобальные, которые вам не будут нужны для дальнейшей работы кода. Это поможет compact освободить больше.

    Важно: не гарантируется, что будет освобождена абсолютно вся неиспользуемая память.




    Crypt. Хеширование паролей

    ^math:crypt[password;salt]

    Хеширует password с учетом salt.

    Параметры:
    ·password - исходная строка;
    ·salt - строка, определяющая алгоритм хеширования и вносящая элемент случайности в результат хеширования, состоит из начала и тела. Начало определяет алгоритм хеширования, тело вносит элемент случайности. Если тело не будет указано, Parser сформирует случайное.

    Неразумно хранить пароли пользователей, просто записывая их на диск или в базу данных - ведь если предположить, что злоумышленник украдет файл или таблицу с паролями, он легко сможет ими воспользоваться. Поэтому принято хранить не пароли, а их хеши - результат надежного однозначного необратимого преобразования строки пароля. Для проверки введенного пароля к нему применяют то же преобразование, передавая в качестве salt сохраненный хеш, а результат сверяют с сохраненным.

    Вносить элемент случайности необходимо, поскольку иначе потенциальный злоумышленник может заранее сформировать таблицу хешей многих частоиспользуемых паролей. Вторая причина: элемент случайности вносится на начальном этапе алгоритма хеширования, что существенно осложняет подбор пароля даже при использовании специальных аппаратных ускорителей.

    Внимание: обязательно задавайте случайное тело salt, или позвольте Parser сделать это за вас, попросту не указывая тело salt, указывая только начало salt.

    Таблица доступных алгоритмов:
    Алгоритм
    Описание
    Начало salt
    Тело "salt"
    MD5
    встроен в Parser, доступен на всех платформах
    $apr1$
    до 8 случайных букв (в любом регистре) или цифр
    MD5
    если поддерживается операционной системой (UNIX)
    $1$
    до 8 случайных букв (в любом регистре) или цифр
    DES
    если поддерживается операционной системой (UNIX)
    (нет)
    2 случайных буквы (в любом регистре) или цифры
    другие
    какие поддерживаются операционной системой (UNIX)
    читайте документацию на вашу операционную систему, функция crypt
    читайте документацию на вашу операционную систему, функция crypt

    Внимание: в Parser для использования в тексте символа '$' его необходимо предварить символом '^'.

    Примечание: Веб-сервер Apache допускает в файлах с паролями (.htpasswd) использовать хеши сформированные по любому из алгоритмов, представленных в таблице, включая алгоритм, встроенный в Parser.




    Degrees, radians. Преобразования градусы-радианы

    Методы выполняют преобразования из градусов в радианы и обратно.

    ^math:degrees(числорадиан)
    - возвращает число градусов, соответствующее заданному числу радиан

    ^math:radians(число градусов)
    - возвращает число радиан, соответствующее заданному числу градусов




    Exp, log. Логарифмические функции

    ^math:exp(число)
    - экспонента по основанию e
    ^math:log(число)
    - натуральный логарифм
    Методы вычисляют значения логарифмических функций от заданного числа

    Примечание (если вы совсем забыли родную школу):
    логарифм по произвольному основанию base вычисляется как log(число)/log(base).




    Класс mail

    Класс предназначен для работы с электронной почтой. Описание настройки Parser для работы этого класса см. Конфигурационный метод.




    Класс MAIN, обработка запроса

    Parser обрабатывает запрошенный документ следующим образом:

    1.
    Считываются, компилируются и инициализируются Конфигурационный файл; затем все файлы с именем auto.p, поиск которых производится начиная от корня веб-пространства, и ниже по структуре каталогов, вплоть до каталога с запрошенным документом; наконец, сам запрошенный документ.
    Все вместе они составляют определение класса MAIN.
    Инициализация заключается в вызове метода auto каждого загруженного файла. Если определение этого метода содержит параметр, при вызове в нем будет передано имя загруженного файла.
    Примечание: результат работы метода посетителю не выводится.

    2.
    Затем вызывается без параметров метод main класса MAIN.
    Т.е. в любом из перечисленных файлов может быть определен метод main, и будет вызван тот, который был определен последним - скажем, определение этого метода в запрошенном документе перекрывает все остальные возможные определения, и будет вызван именно он.
    Результат работы этого метода будет передан пользователю, если не был определен метод postprocess.
    Если в файле не определен ни один метод, то все его тело считается определением метода main.
    Примечание: задание $response:body[нестандартногоответа] переопределяет текст, получаемый пользователем.

    3.
    Если в классе MAIN определен метод postprocess, то результат работы метода main передается единственным параметром этому методу, и пользователь получит уже его результат работы.
    Таким образом вы получаете возможность «дополнительной полировки» результата работы вашего кода.


    Класс math

    Класс math не имеет конструкторов для создания объектов, он обладает только статическими методами и предназначен для вычисления математических выражений. При работе с этим классом необходимо учитывать ограничения на разрядность для класса double.




    Класс memory [3.1]

    Класс предназначен для работы с памятью Parser.
    Его использование поможет вам экономить память в ваших скриптах.

    Для любознательных: в Parser используется известный и хорошо зарекомендовавший себя консервативный сборщик мусора Boehm-Demers-Weiser, см. http://www.hpl.hp.com/personal/Hans_Boehm/gc/.




    Md5. MD5-отпечаток строки [3.0.8]


    ^math:md5[строка]

    Из переданной строки получает «отпечаток» размером 16 байт.
    Выдает его представление в виде строки - байты представлены в шестнадцатеричном виде без разделителей, в нижнем регистре.

    Считается, что практически невозможно
    ·создать две строки, имеющие одинаковый «отпечаток»;
    ·восстановить исходную строку по ее «отпечатку».




    Pow. Возведение числа в степень

    ^math:pow(число;степень)

    Возводит число в степень.


    получаем 90

    ^math:degrees($math:PI/2)
    - получаем 90 (градусов)
    ^math:radians(180)
    - получаем p

    ^math:abs

    ^math:abs(-15.506)
    - получаем 15.506
    ^math:sign(-15.506)
    - получаем -1

    однозначное соответствие имени строке запроса,

    В качестве имени cache-файла возьмем «отпечаток» строки $request:uri, это обеспечит взаимно- однозначное соответствие имени строке запроса, а также избавит нас от необходимости укорачивать строку запроса и очищать ее от спецсимволов.

    ^cache[$cache_directory/^math:md5[$request:uri]]($cache_time){

    }

    Подробная информация о MD5 доступна здесь: http://www.ietf.org/rfc/rfc1321.txt

    получаем 1024

    ^math:pow(2;10) - получаем 1024 ( 210 =1024 )

    Получим случайное число из диапазона

    ^math:random(1000)

    Получим случайное число из диапазона от 0 до 999.

    получаем 45

    ^math:round(45.50)
    - получаем 46
    ^math:floor(45.60)
    - получаем 45
    ^math:ceiling(45.20)
    - получаем 46
    ^math:round(-4.5)
    - получаем -4
    ^math:floor(-4.6)
    - получаем -5
    ^math:ceiling(-4.20)
    - получаем -4

    ^math:cos

    ^math:cos(^math:radians(180)) - получаем -1 (cos p = -1).

    получаем 85

    ^math:trunc(85.506)
    - получаем 85
    ^math:frac(85.506)
    - получаем 0.506

    В разных филиалах компании собираются

    В разных филиалах компании собираются заказы, которые периодически отправляются в центральный офис. Чтобы обеспечить уникальность идентификатора заказа используем UUID.

    # в разных филиалах происходит наполнение таблицы orders и order_details

    # создаем уникальный идентификатор
    $order_uuid[^math:uuid[]]

    # добавляем запись о заказе
    ^void:sql{
    insert into orders
    (order_uuid, date_ordered, total)
    values
    ('$order_uuid', '$date_ordered', $total)
    }
    #цикл по заказанным продуктам вокруг добавления записи о продукте
    ^void:sql{
    insert into order_details
    (order_uuid, item_id, price)
    values
    ('$order_uuid', $item_id, $price)
    }

    # с какой-то периодичностью выбирается часть таблицы orders (и order_details)
    # отправляется (^mail:send[…]) в центральный офис,
    # где части таблиц попадают в общие таблицы orders и order_details
    # БЕЗ проблем с повторяющимся order_id

    Примечание: Parser создает UUID основываясь на случайных числах, а не времени. Параметры:
    ·variant = DCE;
    ·version = DCE Security version, with embedded POSIX UIDs.
    В UUID не все биты случайны, и это так и должно быть:
    xxxxxxxx-xxxx-4xxx-{8,9,A,B}xxx-xxxxxxxxxxxx

    Подробная информация о UUID доступна здесь: http://www.opengroup.org/onlinepubs/9629399/apdxa.htm

    если вы совсем забыли родную

    ^math:sqrt(16) - получаем 4

    Примечание [ если вы совсем забыли родную школу]:
    корень n-ной степени вычисляется как возведение в степень 1/n.

    Пример проверки пароля

    $right[123]
    $from_user[123]
    $crypted[^math:crypt[$right;^$apr1^$]]
    #обратите внимание на то, что $crypted при каждом обращении разный
    $crypted


    ^if(^math:crypt[$from_user;$crypted]
    eq $crypted){
    Казнить нельзя, помиловать
    }{
    Казнить, нельзя помиловать
    }

    Подробная информация о MD5 доступна здесь: http://www.ietf.org/rfc/rfc1321.txt




    Пример создания .htpasswd файла

    @main[]
    $users[^
    table::create{name password
    alice xxxxxx
    bob yyyyyy
    }]

    $htpasswd[^table::create[nameless]{}]
    ^users.menu{
    ^htpasswd.append{$users.name:^math:crypt[$users.password;^$apr1^$]}
    }

    ^htpasswd.save[nameless;.htpasswd-parser-test]




    Простой пример

    Добавление такого определения, скажем, в файл auto.p в корне вашего веб-пространства…
    @postprocess[body]
    ^body.match[
    шило][g]{мыло}

    …приведет к замене одних слов на другие в результатах обработки всех страниц.




    Random. Случайное число

    ^math:random(верхняя_граница)

    Метод возвращает случайное число, попадающее в интервал от 0 до заданного числа, не включая заданное.
    Примечание: на некоторых операционных системах выдает псевдослучайное число.




    Round, floor, ceiling. Округления

    ^math:round(число)
    - округление до ближайшего целого
    ^math:floor(число)
    - округление до целого в меньшую сторону
    ^math:ceiling(число)
    - округление до целого в большую сторону

    Методы возвращают округленное значение заданного числа класса double.


    Send. Отправка сообщения по электронной почте

    ^mail:send[сообщение]

    Метод отправляет сообщение на заданный адрес электронной почты. Можно указать несколько адресов через запятую.
    Пример:
    ^mail:send[
    $.from[
    Вася ]
    $.to[
    Петя ]
    $.subject[
    как дела]
    $.text[
    Как у тебя дела? У меня - изумительно!]
    ]
    В результате будет отправлено сообщение для petya@hotmail.ru с содержимым "Как у тебя дела? У меня - изумительно!".

    сообщение - хеш, в котором могут быть заданы такие ключи:
    ·поле_заголовка
    ·text
    ·html
    ·file
    ·charset
    ·options [3.1.2]

    Внимание: рекомендуется в поле заголовка errors-to задавать адрес, на который может придти сообщение об ошибке доставки письма. По-умолчанию «postmaster».

    charset
    - если задан этот ключ, то заголовок и текстовые блоки сообщения будут перекодированы в указанную кодировку. По умолчанию сообщение отправляется в кодировке, заданной в $request:charset (т.е. не перекодируется).

    Пример:
    $.charset[koi8-r]

    options
    - эти опции будут переданы в командную строку программе sendmail (только под UNIX).

    Также можно задать все поля заголовка сообщения, передав их значение в таком виде (короткая форма):
    $.поле_заголовка[строка]
    или с параметрами (полная форма):
    $.поле_заголовка[
    $.value[
    строка]
    $.
    параметр[строка]
    ]

    Примеры:
    $.from[Вася ]
    $.to[
    Петя ]
    $.subject[Как у тебя дела? У меня - изумительно!]
    $.x-mailer[Parser 3]

    Кроме заголовка можно передать один или оба текстовых блока: text, html. А также любое количество блоков file и message (см. ниже).
    Если будет передано оба текстовых блока, будет сформирована секция MULTIPART/ALTERNATIVE, при прочтении полученного сообщения современные почтовые клиенты покажут HTML, а устаревшие - простой текст.
    Короткая форма:
    $.text[строка]
    Полная форма…
    $.text[
    $.value[
    строка]
    $
    .поле_заголовка[значение]
    ]

    …где value - значение тестового блока, и можно задать все поля заголовка сообщения, передав их как и в хеше сообщение (см. выше).
    Внимание: можно не передавать заголовок content-type, он будет сформирован автоматически. Этот заголовок не влияет на перекодирование, а влияет только на ту кодировку, в которой почтовый клиент будет отображать сообщение.

    Отправка HTML. Короткая форма:
    $.html{строка}
    Полная форма:
    $.html[
    $.value{
    строка}
    $.
    поле_заголовка[значение]
    ]

    Фигурные скобки нужны для переключения вида преобразования по умолчанию на HTML.

    Вложение файла. Короткая форма:
    $.file[файл]
    Полная форма:
    $.file[
    $.value[
    файл]
    $.name[имя_файла]
    ]
    Файл
    - объект класса file, который будет прикреплен к сообщению. MIME-тип данных (content-type заголовок части) определяется по таблице MIME-TYPES (см. Конфигурационный метод).
    Имя_файла - имя, под которым файл будет передан.
    Файл будет передан в uuencode форме.

    Вложение сообщения:
    $.message[сообщение]
    Формат сообщения такой же, как у параметра всего метода.

    Вложений может быть несколько, для чего после имени следует добавить целое число. Пример:
    $.file
    $.file
    2
    $.message
    $.message
    2

    Пример с альтернативными блоками и вложениями:
    ^mail:send[
    $.from[
    Вася ]
    $.to[
    Петя ]
    $.subject[
    как дела]
    $.text[
    Как у тебя дела? У меня изумительно!]
    $.html{
    Как у тебя дела? У меня изумительно!}
    $.file[^
    file::load[binary;perfect_life1.jpg]]
    $.file2[
    $.value[^file::load[binary;perfect_life2.jpg]]
    $.name[
    изумительная_жизнь2.jpg]
    ]
    ]
    В результате будет отослано сообщение для petya@hotmail.ru с содержимым «Как у тебя дела? У меня - изумительно!» в простом тексте и HTML. К сообщению будут приложены две подтверждающие фотографии, на которых…




    Sin, asin, cos, acos, tan, atan. Тригонометрические функции

    ^math:sin(радианы)
    - арксинус
    ^math:asin(число)
    - синус
    ^math:cos(радианы)
    - косинус
    ^math:acos(число)
    - аркосинус
    ^math:tan(радианы)
    - тангенс
    ^math:atan(число)
    - арктангенс

    Методы вычисляют значения тригонометрических функций от заданного числа.


    Sqrt. Квадратный корень числа

    ^math:sqrt(число)

    Вычисляет квадратный корень числа.


    Trunc, frac. Операции с целой/дробной частью числа

    ^math:trunc(число)
    - возвращает целую часть числа
    ^math:frac(число)
    - возвращает дробную часть числа




    Uid64. 64-битный уникальный идентификатор [3.0.8]

    ^math:uid64[]

    Выдает случайную строку вида…
    BA39BAB6340BE370

    Примечание: на некоторых операционных системах выдает псевдослучайную строку.

    См. ^math:uuid[].




    Uuid. Универсальный уникальный идентификатор [3.0.8]

    ^math:uuid[]

    Выдает случайную строку вида…
    22C0983C-E26E-4169-BD07-77ECE9405BA5

    Примечание: на некоторых операционных системах выдает псевдослучайную строку.

    Удобно использовать, когда трудно обеспечить или вообще нецелесообразно использовать сквозную нумерацию объектов.
    Например, при распределенных вычислениях.

    UUID также известен как GUID.




    А что это вообще такое?

    Cache. Сохранение результатов работы кода

    ^cache[файл](числосекунд){код}
    ^cache[файл](число секунд){код}{обработчик проблем}
    [3.1.2]
    ^cache[файл][дата устаревания]{код}
    ^cache[файл][дата устаревания]{код}{обработчик проблем}
    [3.1.2]

    Оператор сохраняет строку, которая получится в результате работы кода. При последующих вызовах обычно происходит считывание ранее сохраненного результата, вместо повторного вычисления, что сильно экономит время обработки запроса и снижает нагрузку на ваши сервера.

    Крайне рекомендуется подключать модули (^use[…]) изнутри кода оператора cache, а не делать это статически (@USE).
    По-возможности, работайте с базой данных (^connect[…]) также внутри кода оператора cache - вы существенно снизите нагрузку на ваш SQL-сервер и повысите производительность ваших сайтов.

    Файл - имя файла-кеша. Если такой файл существует и не устарел, то его содержимое выдается клиенту, если не существует - выполняется код, и результат сохраняется в файл с указанным именем.

    Число секунд - время хранения результата работы кода в секундах. Если это число равно нулю, то результат не сохраняется, а файл с предыдущем сохраненным результатом уничтожается.

    Дата устаревания - время, до которого хранится результата работы кода. Если она в прошлом, то результат не сохраняется, а файл с предыдущим сохраненным результатом уничтожается.

    Код - код, результат которого будет сохранен.

    Обработчик проблем - здесь можно обработать проблему, если она возникнет в коде. В этом отношении оператор похож на try, см. раздел «Обработка ошибок». В отличие от try, можно задать $exception.handled[cache] - это дает указание Parser обработать ошибку особенным образом: достать из файла ранее сохраненный результат работы кода, проигнорировав тот факт, что этот результат устарел.

    Внутри кода допустимы команды, изменяющие время хранения результата работы кода:

    ^cache(число секунд)
    ^cache[дата устаревания]

    Берется минимальное время хранения кода.


    Connect. Подключение к базе данных

    ^connect[строкаподключения]{код}

    Оператор connect осуществляет подключение к серверу баз. Код оператора обрабатывается Parser, работая с базой данных в рамках установленного подключения.

    Parser (в виде модуля к Apache или IIS) кеширует соединения с SQL-серверами, и повторный запрос на соединение с той же строкой подключения не производится, а соединение берется из кеша, если оно еще действительно.

    Вариант CGI также кеширует соединение, но только на один запрос, поэтому явно допустимы конструкции вида:

    ^connect[строка подключения]{…}
    ^connect[строка подключения]{…}

    При этом не будет двух соединений, и это полезно, когда, скажем, изредка соединение нужно, и заранее неизвестно нужно или нет - заранее его можно не делать, а делать визуально многократно, зная, что оно фактически не разрывается.

    Передать SQL-запрос БД может один из следующих методов или конструкторов языка Parser:

    table::sql
    string:sql
    void:sql
    hash::sql
    int:sql
    double:sql

    Замечание: для работы оператора connect необходимо наличие настроенного драйвера баз данных (см. раздел Настройка).

    Форматы строки соединения для поддерживаемых серверов баз данных описаны в приложении.


    Def. Проверка определенности объекта

    Оператор возвращает булевое значение (истина/ложь) и отвечает на вопрос «определен ли объект?» Проверяемым объектом может любой объект Parser: таблица, строка, файл, объект пользовательского класса и т.д.

    defобъект
    не определенными (не def) считаются пустая строка, пустая таблица, пустой хеш и код.


    Eval. Вычисление математических выражений

    ^eval(математическоевыражение)
    ^eval(математическое выражение)[форматная строка]

    Оператор eval вычисляет математическое выражение и позволяет вывести результат в нужном виде, задаваемом форматной строкой (см. Форматные строки).


    -F и -d. Проверка существования файла и каталога

    -f имя_файла - проверка существования файла на диске.
    -dимя_каталога - проверка существования каталога на диске.

    Возвращает результат "истина/ложь" в зависимости от того, существует ли указанный файл или каталог по заданному пути.


    For. Цикл с заданным числом повторов

    ^for[счетчик](от;до){тело}
    ^for[счетчик](от;до){тело}[разделитель]
    ^for[счетчик](от;до){тело}{разделитель}

    Оператор for повторяет тело цикла, перебирая значения счетчика от начального значения до конечного. С каждым выполнением тела значение счетчика автоматически увеличивается на 1.

    Счетчик - имя переменной, которая является счетчиком цикла

    От и до - начальное и конечное значения счетчика, математические выражения, задающие соответственно начало и конец диапазона значений, принимаемых счетчиком. Если конечное значение счетчика меньше начального, тело цикла не выполнится ни разу

    Разделитель - код, который выполняется перед каждым непустым не первым телом

    Замечание: поскольку имена счетчиков могут повторяться, полезно объявлять их локальными переменными метода, где используется цикл for.


    If. Выбор одного варианта из двух

    ^if(логическое выражение){код, если значение логического выражения «истина»}
    ^if(логическое выражение){
    код, если значение логического выражения «истина»
    }{
    код, если значение логического выражения «ложь»
    }


    Оператор вычисляет значение логического выражения. Затем, в зависимости от полученного результата, либо выполняется или не выполняется код (первый вариант использования оператора if), либо для исполнения выбирается код, соответствующий полученному значению логического выражения (второй вариант). На код не накладывается никаких ограничений, в том числе внутри него может содержаться еще один или несколько операторов if.

    Внимание: настоятельно советуем задавать комментарии к частям сложного логического выражения (см. «Комментарии к частям выражения»).




    In. Проверка, находится ли документ в каталоге

    in"/каталог/"

    Возвращает результат "истина/ложь" в зависимости от того, находится ли текщий документ в указанном каталоге.


    Is. Проверка типа

    объектis тип

    Возвращает результат "истина/ложь" в зависимости от того, относится ли левый операнд к заданному типу.
    Полезно использовать этот оператор в случае, если переменная может содержать единственное значение или набор значений (хеш), а также для проверки определенности методов.

    Тип - имя типа, им может быть системное имя (hash, junction, …), или имя пользовательского класса.


    Изменение времени хранения

    ^cache[/data/cache/test2](5){
    по ходу работы вы выяснили,
    что страницу сохранять не нужно:
    ^cache(0)
    }





    Комментарии к частям выражения

    Допустимо использование комментариев к частям математического выражения, которые начинаются со знака #, и продолжаются до конца строки исходного файла или до конца выражения.


    Оператор include

    @include[filename][file]
    $file[^
    file::load[text;$filename]]
    ^process[$
    caller.self]{^taint[as-is][$file.text]}[
    $.file[$filename]
    ]

    Код загружает указанный файл и выполняет его в контексте вызова include. Опция file позволяет указать имя файла, откуда был загружен код. Если возникнет ошибка, вы увидите это «имя файла».


    ^cache

    ^cache[/data/cache/test1](5){
    Нажимайте reload, меняется каждые 5 секунд: ^math:random(100)
    }

    ^connect

    ^connect[mysql://admin:pwd@localhost/p3test]{
    $news[^table::sql{select * from news}]
    }

    определен ли метод» используйте оператор

    $str[Это определенная строка]
    ^if(def $str){
    Строка определена
    }{
    Строка не определена
    }

    Важно: для проверки «содержит ли переменная код» и « определен ли метод» используйте оператор is, а не def. Таким образом. ^if(def $hash.delete){-}{в hash нет элемента «delete»}.

    к частям сложного математического выражения

    ^eval(100/6)[%.2f]
    вернет: 16.67.

    Внимание: настоятельно советуем задавать комментарии к частям сложного математического выражения (см. «Комментарии к частям выражения»).

    к частям сложного математического выражения.

    ^if(
    $age>=$MINIMUM_AGE # не слишком молод
    && $age<=$MAXIMUM_AGE # и не слишком стар
    ){
    Годен
    }

    Внимание: настоятельно советуем задавать комментарии к частям сложного математического выражения. Бывает, что даже вам самим через какое-то время бывает трудно в них разобраться.

    Пример выводит ссылки на недели

    ^for[week](1;4){
    Новости за неделю №$week
    }[
    ]

    Пример выводит ссылки на недели с первой по четвертую, после очередной строки ставится тег перевода строки.

    ^if(in "/news/")

    ^if(in "/news/"){
    Мы в разделе новостей
    }{
    Новости
    }

    ^if(-f "/index.html")

    ^if(-f "/index.html"){
    Главная страница существует
    }{
    Главная страница не существует
    }

    ^switch[$color]

    ^switch[$color]{
    ^case[red]{Необходимо остановиться и подумать о вечном…}
    ^case[yellow]{Настало время собраться и приготовиться!}
    ^case[green]{Покажи им, кто король дороги!}
    ^case[DEFAULT]{Вы дальтоник, или это вовсе не светофор.}
    }

    Обращаем ваше внимание на то,

    @method[command]
    ^switch[$command]{
    ^case[add]{
    добавляем…
    }
    ^case[delete]{
    удаляем…
    }
    ^case[DEFAULT]{
    ^throw[bad.command;$command;Wrong command $command, good are add&delete]
    }
    }

    @main[]
    $action[format c:]
    ^try{
    ^method[$action]
    }{
    ^if($exception.type eq bad.command){
    $exception.handled(1)
    Неправильная команда '$exception.source', задана
    в файле $exception.file, в $exception.lineno строке.
    }
    }

    Результатом работы примера будет
    Неправильная команда 'format c:', задана
    в файле c:/parser3tests/www/htdocs/throw.html, в 15 строке.

    Обращаем ваше внимание на то, что пользователи вашего сайта не должны увидеть технические подробности в сообщениях об ошибках, тем более содержащие пути к файлам, это некрасиво и ненадежно.
    Вывод $exception.file дан в качестве примера и настоятельно не рекомендуется к использованию на промышленных серверах - только для отладки.

    в переменной town, будут преобразованы

    Пример ^^taint.

    $town[Москва]
    $town

    В результате данные, хранящиеся в переменной town, будут преобразованы к типу URI. Русские буквы будут заменены на шестнадцатеричные коды символов и представлены в виде %ХХ.

    где вам необходимы дополнительные операторы,

    Поместите этот код…
    @default[a;b]
    ^if(def $a){$a}{$b}

    … в файл operators.p, в корень вашего веб-сайта.

    Там, где вам необходимы дополнительные операторы, подключите этот модуль. Например, в корневом auto.p, напишите…
    @USE
    /operators.p

    …теперь не только в любой странице, но, что главное, в любом вашем классе можно будет воспользоваться конструкцией
    ^default[$form:name;Аноним]

    Подробности в разделе Создание методов и пользовательских операторов.

    Process. Компиляция и исполнение строки

    ^process{строка}
    ^process[контекст]{строка}
    ^process[контекст]{строка}[опции]
    [3.1.2]

    Строка будет скомпилирована и выполнена как код на Parser, в указанном контексте, или в текущем контексте.
    В качестве контекста можно указать объект или класс.

    Удобно использовать, если какие-то части кода или собственные методы необходимо хранить не в файлах .html, которые обрабатываются Parser, а в каких-то других, или базе данных.

    Также можно указать ряд опций (хеш):
    ·$.main[новое имя для метода main, описанного в коде строки}
    ·$.file[имя файла, из которого взята данная строка}
    ·$.lineno(номер строки в файле, откуда взята данная строка. может быть отрицательным)




    Простая проверка типа

    @main[]
    $date[
    1999-10-10]
    #$date[^date::now[]]
    ^if($date is string){
    ^parse[$date]
    }{
    ^print_date[$date.
    year;$date.month;$date.day]
    }

    @parse[date_string][date_parts]
    $date_parts[^date_string.
    match[(\d{4})-(\d{2})-(\d{2})][]]
    ^print_date[$date_parts.
    1;$date_parts.2;$date_parts.3]

    @print_date[year;month;day]
    Работаем с датой:

    День:
    $day

    Месяц:
    $month

    Год:
    $year


    В этом примере в зависимости от типа переменной $date либо выполняется синтаксический анализ строки, либо методу print_date передаются поля объекта класса date:


    Простые примеры

    ^process{@extra[]
    Здоровья прежде всего…
    }
    Метод extra будет добавлен к текущему классу, и его можно будет вызывать в дальнейшем.

    ^process[$engine:CLASS]{@start[]
    Мотор…
    }
    Метод start будет добавлен к пользовательскому классу engine.

    $running_man[^man::create[Вася]]
    ^process[$running_man]{
    Имя: $name

    }
    Код будет выполнен в контексте объекта $running_man, соответственно, может воспользоваться полем name этого объекта, выдаст «Вася».


    Проверка наличия в переменной кода

    Т.к. оператор def не считает переменные, содержащие код, определенными, для проверки наличия в переменной кода необходимо использовать is:
    $something{^do[]}
    ^if($somethingis
    junction){
    #
    выполнение кода
    $something
    }





    Проверка определенности метода

    Значение $имя_метода, это тоже junction, поэтому проверять существование метода необходимо также оператором is, а не def:
    @body[]
    тело

    @
    main[]
    Старт
    ^if($body is junction){
    ^body[]
    }{
    Метод «body» не определен!
    }
    Финиш




    Rem. Вставка комментария

    ^rem{комментарий}

    Весь код, содержащийся внутри оператора, не будет выполнен. Используется для комментирования блоков кода.




    Системное поле объекта: CLASS

    $объект.CLASS - хранит ссылку на класс объекта.

    Это необходимо при задании контекста компиляции кода (см. «process. Компиляция и исполнение строки».




    Сложный пример

    Часто удобно поместить компилируемый код в некоторый метод с именем, вычисляющимся по ходу работы:
    # это исходный код, обратите внимание, на ^^
    $
    source_code[2*2=^^eval(2*2)]
    # по ходу работы выясняется, что необходимо создать метод с именем method1
    $method_name[method1]
    #
    компилируем исходный код, помещяем его в новый метод
    ^process{$
    source_code}[
    $.main[$method_name]
    ]

    # далее по коду можно вызывать метод method1
    ^method1[]

    Данный пример будет продолжать работать, даже если в $source_code будет определен ряд методов, поскольку опция main задает новое имя методу main.




    Создание объекта

    ^класс::конструктор[параметры]

    Конструктор создает объект класса, наделяя его полями и методами класса. Параметры конструкторов подробно описаны в соответствующем разделе.
    Примечание: результат работы конструктора - созданный объект, обычный результат работы метода игнорируется (никуда не попадает).


    Switch. Выбор одного варианта из нескольких

    ^switch[строкадля сравнения]{
    ^case[вариант1]{действие для 1}
    ^case[вариант2]{действие для 2}
    ^case[вариант3;вариант 4]{действие для 3 или 4}

    ^case[DEFAULT]{вариант по умолчанию}
    }

    ^switch(математическое выражение){
    ^case(вариант1){действие для 1}
    ^case(вариант2){действие для 2}
    ^case(вариант3;вариант 4){действие для 3 или 4}

    ^case[DEFAULT]{вариант по умолчанию}
    }

    Оператор switch сравнивает строку или результат математического выражения со значениями, перечисленными в case. В случае совпадения выполняется код, соответствующий совпавшему значению. Если совпадений нет, выполняется код, соответствующий значению DEFAULT (пишется только заглавными буквами).

    Если код для DEFAULT не определен и нет совпадений со значениями, перечисленными в case, ни один из вариантов кода, присутствующих в операторе switch, выполнен не будет.


    Таблицы преобразований

    as-is
    изменений в тексте не делается

    file-spec
    символы * ? ' " < > |
    преобразуются в "_XX",
    где XX - код символа в шестнадцатеричной форме

    uri
    символы за исключением цифр, строчных и прописных латинских букв, а также следующих символов: _ - . "
    преобразуется в %XX где XX - код символа в шестнадцатеричной форме

    http-header
    то же, что и URI

    mail-header
    если известен charset (если неизвестен, не будут работать up/low case), то фрагмент, начиная с первой буквы с восьмым битом и до конца строки, будет представлен в подобном виде:
    Subject: Re: parser3: =?koi8-r?Q?=D3=C5=CD=C9=CE=C1=D2?=

    sql
    в зависимости от SQL-сервера
    для Oracle/ODBC меняется ' на ''
    для PgSQL меняется ' на '' и \ на \\
    для MySQL делается средствами самого MySQL

    js
    " преобразуется в \"
    ' преобразуется в \'
    \ преобразуется в \\
    символ конца строки преобразуется в \n
    символ с кодом 0xFF предваряется \

    xml
    & преобразуется в &
    > преобразуется в >
    < преобразуется в <
    " преобразуется в "
    ' преобразуется в '

    html
    & преобразуется в &
    > преобразуется в >
    < преобразуется в <
    " преобразуется в "

    optimized-as-is
    optimized-xml
    optimized-html

    дополнительно к заменам выполняется оптимизация по "white spaces" (символы пробела, табуляция, перевода строки).

    Идущие подряд перечисленные символы заменяются только одним, который встречается в коде первым
    Ряд taint преобразований Parser делает автоматически, так, имена и пути файлов всегда автоматически file-spec преобразуются, и когда вы пишите…

    ^file::load[filename]

    …Parser выполняет…

    ^file::load[^taint[file-spec][filename]]

    Аналогично, при задании HTTP-заголовков и заголовков писем, происходят http-header и mail-header преобразования соответственно. А при DOM-операциях текстовые параметры всех методов автоматически xml преобразуются.

    Также Parser выполняет ряд автоматических untaint преобразований:
    вид
    что преобразуется
    sql
    тело SQL-запроса
    xml
    XML код при создании объекта класса xdoc




    Throw. Сообщение об ошибке

    ^throw[type;source]
    ^throw[type;source;comment]

    Оператор throw сообщает об ошибке типа type, произошедшей по вине source, с комментарием comment.

    Эта ошибка может быть перехвачена и обработана при помощи оператора try.

    Не перехватывайте ошибки только для их красивого вывода, пусть этим централизованно займется метод unhandled_exception, вызываемый Parser если ни одного обработчика ошибки так и не будет найдено. Кроме прочего, произойдет запись в журнал ошибок веб-сервера, который можно регулярно просматривать на предмет имевших место проблем.


    Try. Перехват и обработка ошибок

    ^try{код,ошибки которого попадают…}{…в этот обработчик в виде $exception}

    Если по ходу работы кода возникла ошибка, создается переменная $exception, и управление передается обработчику.

    $exception, это такой hash:

    $exception.type
    строка, тип ошибки.
    Определен ряд системных типов, также тип ошибки может быть задан в операторе throw.
    $exception.source
    строка, источник ошибки (ошибочное имя файла, метода, …)
    $exception.file
    $exception.lineno
    $exception.colno

    файл, содержащий source,
    номера строки и колонки в нем
    $exception.comment
    комментарий к ошибке, по английски
    $exception.handled
    истина или ложь, флаг «обработана ли ошибка»
    необходимо зажечь этот флаг в обработчике, если вы обработали переданную вам ошибку
    Обработчик обязан сообщить Parser, что данную ошибку он обработал, для чего только для нужных типов ошибок он должен зажечь флаг:
    $exception.handled(1)

    Если обработчик не зажег этого флага, ошибка считается необработанной, и передается следующему обработчику, если он есть.

    Если ошибка так и не будет обработана, если есть, вызывается метод unhandled_exception и ему передается информация об ошибке, стек вызовов, приведших к ошибке, и выдаются результаты его работы. А также производится запись в журнал ошибок веб-сервера.



    Untaint, taint. Преобразование данных

    ^untaint{код}
    ^untaint[вид преобразования]{код}
    ^taint[текст]
    ^taint[вид преобразования][текст]


    Оператор taint помечает весь переданный ему текст, как нуждающийся в преобразовании заданного вида. Оператор untaint выполняет переданный ему код и помечает, как нуждающиеся в преобразовании, те части результата выполнения кода, которые не являлись частью кода на Parser, а поступили извне, например, из полей формы.

    Если вид преобразования не указан, оператор untaint делает преобразование вида as-is, а оператор taint помечает текст как tainted (неопределенно «грязный», без указания вида преобразования).

    Само преобразование Parser выполняет позже, при выдаче результата обработки документа пользователю, перед выполнением SQL-запроса или отправкой письма.

    Преобразование заключается в замене одних символов на другие в соответствии с внутренними таблицами пребразований. Предусмотрены следующие виды преобразований:

    as-is
    file-spec
    http-header
    mail-header
    uri
    sql
    js
    xml
    html

    optimized-as-is

    optimized-xml
    optimized-html

    Выполняемые замены для каждого вида преобразования сведены в таблице ниже.

    Важное замечание: «чистый» код также подвергается автоматическому преобразованию. Parser выполняет оптимизацию по «white spaces» (символы пробела, табуляция, перевода строки). Идущие подряд перечисленные символы заменяются только одним, который встречается в коде первым. В общем виде можно представить весь набираемый код следующим образом:

    ^untaint[optimized-html]{здесьнаходится код, набранный скриптовальщиком}

    Необходимо учитывать это при создании кода! Схема автоматического преобразования Parser - optimized-html.

    Пример:

    Пример на ^^untaint.






    $tainted[$form:field]
    "Грязные" данные - $tainted

    "Чистые" данные -
    ^untaint{$tainted}

    В результате выполнения этого кода поступившие через поля формы данные не будут преобразовываться Parser и выдадутся в том виде, в каком они поступили. В квадратных скобках оператора untaint задается вид выполняемого преобразования. Здесь мы опускаем квадратные скобки в методе untaint и используем параметр по умолчанию [as-is].

    Как уже было сказано выше, большую часть работы по преобразованию данных Parser выполняет автоматически. Внешние данные, которые используются в «чистом» коде преобразуются к optimized-html.

    Оператор taint также помечает преобразование данных в соответствии с указанной таблицей, но, в отличие от untaint, используется для «чистых» данных, т.е. созданных кодером, либо поступивших извне и уже обработанных untaint. Еще раз обращаем внимание, что преобразование происходит только в момент выдачи данных во внешнюю среду (пользователю, в файл или базу данных). Если не задан вид преобразования, данные останутся "грязными", но никаких замен сделано не будет.


    Use. Подключение модулей

    ^use[файл]

    Оператор позволяет использовать модуль из указанного файла. Если путь к файлу начинается с "/", то считается, что это путь от корня веб-пространства. В любом другом случае Parser будет искать модуль по путям, определенным в переменной $CLASS_PATH в Конфигурационном методе.

    Для подключения модулей также можно воспользоваться конструкцией:
    @USE
    имя файла 1
    имя файла 2



    Разница между этими конструкциями в том, что использование @USE подключает файлы с модулями до начала выполнения кода, в то время как оператор use может быть вызван непосредственно из тела программы, например:

    ^if(условие){
    ^use[модуль1]
    }{
    ^use[модуль2]
    }

    Замечание: попытки подключить уже подключенные ранее модули не приводят к повторным считываниям файлов с диска. Однако нужно иметь ввиду, что для этого полное имя файла подключаемого модуля должно с точностью до символа совпадать с именем файла уже подключенного ранее модуля. В случае написания ^use[module.p] и ^use[sub/../module.p] будет считаться, что идет попытка подключить разные модули.

    Крайне рекомендуем использовать возможность сохранения результатов работы кода, используя оператор use для подключения необходимых модулей в коде оператора cache.




    Вызов метода

    ^объект.метод[параметры]

    Вызов метода класса, к которому принадлежит объект. Параметры конструкторов подробно описаны в соответствующем разделе.

    Если не указан объект, то конструкция является вызовом метода текущего класса (если у класса нет метода с таким именем, будет вызван метод базового класса) или оператора. При совпадении имен вызывается оператор.

    Методы бывают статические и динамические.

    Динамическийметод - код выполняется в контексте объекта (экземпляра класса).

    Статический метод - код выполняется в контексте самого класса, то есть метод работает не с конкретным объектом класса, а со всем классом (например, классы MAIN, math, mail)


    While. Цикл с условием

    ^while(условие){тело}

    Оператор while повторяет тело цикла, пока условие истинно. Если условие заведомо ложно, тело цикла не выполнится ни разу.



    Значение поля объекта

    $объект.поле

    Получение значения поля объекта.


    А что это вообще такое?

    Использование Parser в качестве интерпретатора скриптов

    /путь/к/parser3 файл_со_скриптом
    x:\путь\к\parser3 файл_со_скриптом

    Выполнять скрипты можно и без веб-сервера, достаточно запустить интерпретатор Parser, передав ему в командной строке параметр - имя скрипта. При этом корнем веб-пространства считается текущий каталог.

    При этом ошибки попадут в стандартный поток ошибок, который можно перенаправить в желаемый файл так:
    команда 2>>error_log
    Внимание: не забывайте его время от времени очищать.

    На UNIX можно также использовать стандартный подход с заданием команды запуска интерпретатора в первой строке скрипта:
    #!/путь/к/parser3
    #ваш код
    Проверка: ^eval(2*2)
    Не забудьте зажечь биты атрибута, разрешающие исполнение владельцу и группе. Команда:
    chmod ug+x файл




    Конфигурационный файл

    Пример файла включен в поставку (см. auto.p.dist).

    Этот файл - основной, с которого начинается сборка класса MAIN. Может содержать Конфигурационный метод, который выполняется первым, до метода auto, и задает важные системные параметры.

    После выполнения конфигурационного метода можно задать кодировку ответа и кодировку, в которой набран код (по умолчанию в обоих случаях используется кодировка UTF-8):

    Рекомендуемый код:
    @auto[]
    #source/client charsets
    $request:charset[windows-1251]
    $response:charset[windows-1251]
    $response:content-type[
    $.value[text/html]
    $.charset[$response:charset]
    ]

    Примечание: для корректной работы методов upper и lower класса string с национальными языками (в том числе русским) необходимо корректное задание $request:charset.

    Также здесь рекомендуется определить путь к классам вашего сайта:
    $CLASS_PATH[/../classes]

    И строку соединения с SQL-сервером, используемым на вашем сайте (пример для ODBC):
    $SQL.connect-string[odbc://DSN=www_mydomain_ru^;UID=user^;PWD=password]
    Примечание: вашем коде вы будете использовать ее так:
    ^connect[$SQL.connect-string]{…}

    Советуем поместить сюда же определение метода unhandled_exception, который будет выводить сообщение о возможных проблемах на вашем сайте.

    Внимание: конечно, Конфигурационный файл можно не использовать, а Конфигурационный метод поместить в файл auto.p в корне веб-пространства, однако в разных местах размещения сервера (например: отладочная версия и основной сервер) конфигурации скорее всего будут различными, и очень удобно, когда эти различия находятся в отдельном файле и вне веб-пространства.




    Конфигурационный метод

    Если в файле определен метод conf,он выполняется первым, до auto, и задает важные системные параметры:
    ·файлы, описывающие кодировки символов,
    ·ограничение на размер HTTP POST-запроса,
    ·сервер/программу отправки почты,
    ·SQL-драйвера и их параметры,
    ·таблицу соответствия расширения имени файла и его mime-типа.
    Рекомендуется поместить этот метод в Конфигурационный файл.

    Определение метода:
    @conf[filespec]
    filespec
    - полное имя файла, содержащего метод.

    Всегда доступна и не нуждается в загрузке файла кодировка UTF-8, являющаяся для Parser кодировкой по умолчанию.
    Чтобы сделать доступными для использования Parser другие кодировки, необходимо указать файлы их описывающие, делается это так:
    $CHARSETS[
    $.windows-1251[/полный/путь/к/windows-1251.cfg]

    ]
    См. Описание формата файла, описывающего кодировку.

    Максимальный размер POST данных:
    $LIMITS[
    $.post_max_size(10*0x400*0x400)
    ]

    Параметр отправки писем (см. ^mail:send[…])…

    …под Windows и UNIX (под UNIX [3.1.2]) адрес SMTP-сервера
    $MAIL[
    $.SMTP[mail.office.design.ru]
    ]

    …под UNIX в safe-mode версиях, настроить программу отправки можно только при сборке Parser из исходных кодов, в бинарных версиях, распространяемых с сайта parser.ru, задана команда
    /usr/sbin/sendmail -i -t -f postmaster
    Только в unsafe-mode версиях можно задать программу отправки почты самому:
    $MAIL[
    $.sendmail[/custom/mail/sending/program params]
    ]
    и, по умолчанию, используется эта…
    /usr/sbin/sendmail -t -i -f postmaster
    …или эта…
    /usr/lib/sendmail -t -i -f postmaster
    …команда, в зависимости от вашей системы.
    При отправке письма вместо «postmaster» будет подставлен адрес отправителя из письма из обязательного поля заголовка «from».

    Также можно задать таблицу SQL-драйверов:
    $SQL[
    $.drivers[^table::create{protocol driver client
    mysql /full/disk/path/parser3mysql.dll /full/disk/path/libmySQL.dll
    odbc /full/disk/path/parser3odbc.dll
    pgsql /full/disk/path/parser3pgsql.dll /full/disk/path/libpq.dll
    oracle /path/to/parser3oracle.dll C:\Oracle\Ora81\BIN\oci.dll?PATH+=^;C:\Oracle\Ora81\bin
    }]
    ]
    В колонке client таблицы drivers допустимы параметры клиентской библиотеке, отделяемые знаком ? от имени файла библиотеки, в таком виде:
    имя1=значение1&имя2=значение2&…
    а также имя+=значение.
    Эти переменные будут занесены(=) или добавлены к имеющемуся значению(+=) в программное окружение (environment) перед инициализацией библиотеки. В частности, удобно добавить путь к Oracle библиотекам здесь, если этого не было сделано в системном программном окружении (system environment).

    Таблица типов файлов:
    #файл, создаваемый ^file::load[…],
    #при выдаче в $response:body задаст этот $response:content-type
    $MIME-TYPES[^table::create{ext mime-type
    zip application/zip
    doc application/msword
    xls application/vnd.ms-excel
    pdf application/pdf
    ppt application/powerpoint
    rtf application/rtf
    gif image/gif
    jpg image/jpeg
    jpeg image/jpeg
    png image/png
    tif image/tiff
    html text/html
    htm text/html
    txt text/plain
    mts application/metastream
    mid audio/midi
    midi audio/midi
    mp3 audio/mpeg
    ram audio/x-pn-realaudio
    rpm audio/x-pn-realaudio-plugin
    ra audio/x-realaudio
    wav audio/x-wav
    au audio/basic
    mpg video/mpeg
    avi video/x-msvideo
    mov video/quicktime
    swf application/x-shockwave-flash
    }]

    Расширения имен файлов в таблице должны быть написаны в нижнем регистре. Поиск по таблице нечувствителен к регистру, т.е. файл FACE.GIF получит mime-тип image/gif.




    Передача параметров

    Параметры могут передаваться в разных скобках и, соответственно, будут по-разному обрабатываться:

    (выражение)
    - вычисление параметра происходит при каждом обращении к нему внутри вызова метода
    [код]
    - вычисление параметра происходит один раз перед вызовом метода
    {код}
    - вычисление происходит при каждом обращении к параметру внутри вызова метода
    Пример на различие скобок, в которых передаются параметры:
    @main[]
    $a(20)
    $b(10)
    ^sum[^
    eval($a+$b)]


    ^sum{^eval($a+$b)}

    @sum[c]
    ^
    for[b](100;110){
    $c
    }[

    ]

    Здесь хорошо видно, что в первом случае код был вычислен один раз перед вызовом метода sum, и методу передался результат кода - число 30. Во втором случае вычисление кода происходило при каждом обращении к параметру, поэтому результат менялся в зависимости от значения счетчика.

    Параметров может быть сколь угодно много или не быть совсем. Если в однотипных скобках несколько параметров, то они могут отделяться друг от друга точкой с запятой. Допустимы любые комбинации различных типов параметров.

    Например…
    ^if(условие){когда да;когда нет}
    …эквивалентно…
    ^if(условие){когда да}{когда нет}




    Системные ошибки

    type
    Пример возникновения
    Описание
    parser.compile
    ^test[}
    Ошибка компиляции кода. Непарная скобка, и т.п.
    parser.runtime
    ^if(0).
    Методу передано неправильное количество параметров или не тех типов, и т.п.
    parser.interrupted
    Загрузка страницы прервалась (пользователь остановил загрузку страницы или истекло время ожидания)
    number.zerodivision
    ^eval(1/0), ^eval(1\0) или ^eval(1%0)
    Деление или остаток от деления на ноль
    number.format
    ^eval(abc*5)
    Преобразование к числу нечисловых данных
    file.missing
    ^file:delete[skdfjs.delme]
    файл отсутствует
    file.access
    ^table::load[.]
    Нет доступа к файлу
    image.format
    ^image::measure[index.html]
    Файл изображения имеет неправильный формат (возможно, расширение имени не соответствует содержимому, или файл пуст?)
    sql.connect
    ^connect[mysql://baduser:pass@host/db]{}
    Сервер баз данных не может быть найден или временно недоступен
    sql.execute
    ^void:sql{bad select}
    Ошибка исполнения SQL запроса
    xml
    ^xdoc::create{}
    Ошибочный XML код или операция
    smtp.connect
    SMTP сервер не может быть найден или временно недоступен
    smtp.execute
    Ошибка отправки письма по SMTP протоколу
    email.format
    Ошибка в email адресе: адрес пустой или содержит неправильные символы
    email.send
    Ошибка запуска почтовой программы
    http.host
    ^file::load[http://notfound/there]
    Сервер не найден
    http.connect
    ^file::load[http://not_accepting/there]
    Сервер найден, но не принимает соединение
    http.response
    ^file::load[http://ok/there]
    Сервер найден, соединение принял, но выдал некорректный ответ (нет статуса, заголовка)
    http.status
    ^file::load[http://ok/there]
    Cервер выдал ответ со статусом не равным 200 (не успешное выполнение запроса)
    http.timeout
    Загрузка документа с HTTP-сервера не завершилась в отведенное для нее время





    А что это вообще такое?

    Body. Получение текста запроса

    $request:body

    Получение текста HTTP POST-запроса.

    Вариант использования: можно написать свой XML-RPC сервер (см. http://xmlrpc.org).




    Body. Задание нового тела ответа

    $response:body[DATA]

    Замещает все тело ответа значением DATA.

    DATA - строка (string) или файл (file). Если передан файл с известным content-type (см. поля объекта класса file), этот заголовок передается в пользователю.

    См. также $response:download.


    Charset. Задание кодировки документов на сервере

    $request:charset[кодировка]

    Задает кодировку документов, обрабатываемых на сервере.
    При обработке запроса считается, что в этой кодировке находятся все файлы на сервере.

    По умолчанию используется кодировка UTF-8.

    Список допустимых кодировок определяется Конфигурационным методом.
    Рекомендуется определять кодировку документов в Конфигурационном файле.

    См. также «Задание кодировки ответа».




    Charset. Задание кодировки ответа

    $response:charset[кодировка]

    Задает кодировку ответа.
    После обработки запроса результат перекодируется в эту кодировку.

    По умолчанию используется кодировка UTF-8.

    Список допустимых кодировок определяется Конфигурационным методом.
    Рекомендуется определять кодировку документов в Конфигурационном файле.

    См. также «Задание кодировки документов на сервере».




    Clear. Отмена задания новых заголовков HTTP-ответа

    ^response:clear[]

    Метод отменяет все действия по переопределению полей ответа.




    Document-root. Корень веб-пространства[3.1.2]

    $request:document-root[/дисковый/путь/к/корню/вашего/веб-пространства]

    По-умолчанию, $request:document-root равен значению, которое задается в веб-сервере.
    Однако иногда его удобно заменить.

    См. также «Пути к файлам и каталогам».




    Download. Задание нового тела ответа

    $response:download[DATA]

    Идентичен $response:body, но выставляет флаг, который браузер воспринимает как «Предложить пользователю сохранить файл на диске».

    Браузеры умеют отображать файлы некоторых типов прямо внутри своего окна (например: .doc, .pdf файлы).
    Однако бывает необходимо дать возможность посетителю скачать файл по простому нажатию на ссылку.


    Класс request

    Класс содержит статические поля, которые позволяют получать информацию, передаваемую браузером веб-серверу (по HTTP протоколу).

    Для работы с полями форм (
    ) и строкой после второго ? (/?a=b?thisText) используйте класс form.

    Часть информации о запросе доступна через переменные окружения, см. «Получение значения поля запроса».




    Класс response

    Класс позволяет дополнять стандартные HTTP-ответы сервера. Класс не имеет конструкторов для создания объектов.




    пользователь запросил такую страницу:

    Предположим, пользователь запросил такую страницу: http://www.mysite.ru/news/articles.html?year=2000&month=05&day=27

    Тогда:
    $request:query

    вернет:
    year=2000&month=05&day=27

    пользователь запросил такую страницу:

    Предположим, пользователь запросил такую страницу: http://www.mysite.ru/news/articles.html?year=2000&month=05&day=27

    Тогда:
    $request:uri

    вернет:
    /news/articles.html?year=2000&month=05&day=27

    Пример перенаправления браузера

    $response:location[/]
    $response:refresh[
    $.value[0]
    $.url[/]
    ]




    выдача PDF файла

    Посетитель заходит на страницу с таким HTML…
    Скачать документацию

    download_documentation.html:
    $response:download[^file::load[binary;documentation.pdf]]

    …и нажимает на ссылку, браузер предлагает ему Скачать/Запустить.

    Пример выдачи создаваемой картинки

    $square[^image::create(100;100;0x000000)]
    ^square.circle(50;50;10;0xFFFFFF)
    $response:body[^square.gif[]]

    В браузере будет выведен черный квадрат с белой окружностью. Кроме того, автоматически будет установлен нужный тип файла (content-type) по таблице MIME-TYPES.




    Пример задания заголовка expires в значение «завтра»

    $response:expires[^date::now(+1)]




    Пример замены всего тела на результат работы скрипта

    $response:body[^file::cgi[script.cgi]]

    Заменит весь ответ результатом работы программы script.cgi.





    Query. Получение строки запроса

    $request:query

    Возвращает строку после ? в URI (значение переменной окружения QUERY_STRING).
    Для работы с полями форм () и строкой после второго ? (/?a=b?thisText) используйте класс form.


    Uri. Получение URI страницы

    $request:uri

    Возвращает URI документа.


    Заголовки HTTP-ответа

    $response:поле[значение]
    $response:поле

    Поле соответствует заголовку HTTP-ответа, выдаваемого Parser. Его можно как задавать, так и считывать. Значением может быть дата, строка или хеш с обязательным ключом value. Дата может использоваться и в качестве значения поля и в качестве значения атрибута поля, при этом она будет стандартно отформатирована.

    Примечание: имя поля, при задании значения, преобразуется к нижнему регистру, а при считывании ищется в нижнем регистре.


    А что это вообще такое?

    Format. Вывод числа в заданном формате

    ^строка.format[форматная_строка]

    Метод выводит значение переменной в заданном формате (см. Форматные строки).
    Выполняется автоматическое преобразование строки к числу.


    Int, double. Преобразование строки к числу

    ^строка.int[]
    ^строка.int(значение по умолчанию)
    ^строка.double[]
    ^строка.double(значение по умолчанию)


    Преобразуют значение переменной $строка к целому или вещественному числу соответственно, и возвращает это число.

    Можно задать значение по умолчанию, которое будет получено, если преобразование невозможно или строка пуста или состоит только из "white spaces" (символы пробела, табуляция, перевода строки).

    Значение по умолчанию можно использовать при обработке данных, получаемых интерактивно от пользователей. Это позволит избежать появления текстовых значений в математических выражениях при вводе некорректных данных, например, строки вместо ожидаемого числа.

    Внимание: использование пустой строки в математических выражениях не является ошибкой, ее значение считается нулем.
    Внимание: преобразование строки, не являющейся целым числом к целому числу является ошибкой (пример: строка «1.5» не является целым числом).


    Класс status

    Класс предназначен для анализа текущего состояния скрипта на Parser.
    Его использование поможет вам найти узкие места в ваших скриптах.




    Класс string

    Класс для работы со строками. В выражении строка считается определенной (def), если она не пуста. Если в строке содержится число, то при попытке использовать его в математических выражениях содержимое строки будет автоматически преобразовано к double. Если строка пуста, ее числовое "значение" в математических выражениях считается нулем.

    Создание объекта класса string:
    $str[Строка, которая содержится в объекте]




    Left, right. Подстрока слева и справа

    ^строка.left(N)
    ^строка.right(N)

    Методы возвращают N первых или последних символов строки соответственно. Если длина строки меньше N, то возвращается вся строка.


    Length. Длина строки

    ^строка.length[]

    Возвращает длину строки.


    Match. Поиск подстроки по шаблону

    ^строка.match[шаблон]
    ^строка.match[шаблон][опции поиска]


    Осуществляет поиск в строке по шаблону.
    Шаблон- это регулярное выражение, совместимое с PCRE (Perl compatible regular expressions).
    Частичный перевод описания PCRE приведен в Приложении 4.

    Предусмотрены следующие опции поиска:
    i - не учитывать регистр;
    x - игнорировать символы white space и разрешить #комментарий до конца строки;
    s - символ $ считать концом всего текста (опция по умолчанию);
    m - символ $ считать концом строки, но не всего текста;
    g - найти все вхождения строки (а не только первое);
    ' - вычислять значения столбцов prematch, match, postmatch.

    Поскольку символы ^ и $ используются в Parser, в шаблоне вместо символа ^ используется строка ^^, а вместо символа $ - строка ^$ (см. Литералы).

    Если в шаблоне не используются круглые скобки, то оператор возвращает булевое значение истина при обнаружении подстроки, соответствующей шаблону, или ложь, если совпадений не найдено.

    Если в шаблоне используются круглые скобки, то вместо булевого значения оператор создает таблицу совпадений (объект класса table) со столбцами prematch, match, postmatch, 1, 2,…, n,
    где:
    prematch столбец с подстрокой от начала строки до совпадения
    match столбец с подстрокой, совпавшей с шаблоном
    postmatch столбец с подстрокой, следующей за совпавшей подстрокой до конца строки
    1, 2,…, n столбцы с подстроками, соответствующими фрагментам шаблона, заключенным в круглые скобки, n - номер открывающей круглой скобки

    Если указана опция поиска g, будет создана таблица найденного по шаблону (по одной строке на каждое вхождение).
    При этом не важно, есть в шаблоне круглые скобки или нет.

    Внимание: значения столбцов prematch, match, postmatch вычисляются только если указана опция ' .




    Match. Замена подстроки, соответствующей шаблону

    ^строка.match[шаблон][опциипоиска]{замена}

    Осуществляет поиск в строке по шаблону и производит замену совпавшей подстроки на заданную. Механизм поиска устроен так же, как и у предыдущего метода. Внутри кода замены доступна автоматически создаваемая таблица совпадений match, которая была рассмотрена выше. Дополнительно можно задать еще одну опцию поиска:

    g - задает замену всех совпавших подстрок


    Memory. Информация о памяти под

    Это поле - хеш с информацией о памяти, находящейся под контролем сборщика мусора.

    Поле
    Значение (в килобайтах)
    Детали
    used
    Занято
    В это число не включен размер cлужебных данных самого сборщика мусора.
    free
    Свободно
    Свободная память скорее всего фрагментирована.
    ever_allocated_since_compact
    Было выделено с момента последней сборки мусора. См. memory:compact.
    Между сборками мусора это число только растет.
    Факты освобождения памяти без сборки мусора на него не влияют, только сборки мусора.
    ever_allocated_since_start
    Было выделено за все время обработки запроса
    Это число только растет.
    Ни факты сборки мусора, ни освобождения памяти между сборками мусора на него не влияют.




    Mid. Подстрока с заданной позиции

    ^строка.mid(P;N)
    ^строка.mid(P)


    Возвращает подстроку, которая начинается с позиции P и имеет длину N (если N не задано, то возвращается подстрока с позиции P до конца строки). Отсчет P начинается с нулевой позиции. Если P+N больше длины строки, то будут возвращены все символы строки, начиная с позиции P.


    Pid. Идентификатор процесса

    Идентификатор процесса (process) операционной системы, в котором работает Parser.




    Pos. Получение позиции подстроки

    ^строка.pos[подстрока]

    Возвращает число int - позицию первого символа подстроки в строке (начиная с нуля), или -1, если подстрока не найдена.


    $var

    $var[15.67678678]
    ^var.format[%.2f]

    Возвратит: 15.68

    $str[

    $str[О, сколько нам открытий чудных!…]
    ^str.length[]

    Вернет: 32

    На экран будет выведено:

    $str[О, сколько нам открытий чудных!…]
    ^str.left(10) ^str.right(10)

    На экран будет выведено: О, сколько чудных!…

    ^str.match

    $str[2002.01.01]
    ^str.match[(\d+)\.(\d+)\.(\d+)][g]{Год $match.1, месяц $match.2, число $match.3}

    Выведет: Год 2002, месяц 01, число 01.

    Выведет на экран:

    $str[О, сколько нам открытий чудных!…]
    ^str.mid(3;20)

    Выведет на экран: сколько нам открытий

    ^str.pos

    $str[полигон]
    ^str.pos[гон]

    Вернет: 4

    Выведет на

    $s[A magic moment I'll remember!]
    Исходная строка: $s

    $rep[^table::create{from to
    A An
    magic ugly}]
    Исковерканная строка: ^s.replace[$rep]

    Выведет на экран:
    Исходная строка: A magic moment I'll remember!
    Исковерканная строка: An ugly moment I'll remember!

    запросе insert будет правильно обработан

    Задача: из SQL-сервера А достать данные, положить в SQL-сервер Б.

    Если оба SQL-сервера доступны с какой-то машины, можно так:

    ^connect[А]{
    $data[
    # код, наполняющий data данными из SQL-сервера A
    ]
    ^connect[Б]{
    ^void:sql{insert into table x (x) values ('$data')}
    }
    }

    При этом $data в SQL- запросе insert будет правильно обработан по правилам SQL-диалекта сервера Б.

    Однако если оба SQL-сервера недоступны одновременно с какой-то машины, можно так:

    ^connect[А]{
    $data[
    # код, наполняющий data данными из SQL-сервера A
    ]
    $string[^untaint[sql]{insert into table x (x) values ('$data')}]
    ^connect[локальный фиктивный Б]{
    # это соединение нужно только для того,
    # чтобы задать правила обработки для SQL-диалекта сервера Б
    ^string.save[B-inserts.sql]
    }
    }

    При этом в файл B-inserts.sql запишется правильно обработанный SQL-запрос.

    Используя этот метод, полезно конструировать

    ^string:sql{select name from company where company_id=$company_id}

    Используя этот метод, полезно конструировать SQL-запрос так, чтобы в ответе заведомо содержалась одна строка из одного столбца.

    ^str.upper[]

    $str[Москва]
    ^str.upper[]

    Вернет: МОСКВА.

    Пример горизонтального разбиения

    $str[/a/b/c/d]
    $parts[^str.split[
    /;lh]]
    $parts.0
    , $parts.1, $parts.2

    Выведет:
    , a, b




    Пример отсечения указанных символов

    $path[/section/subsection/]
    ^path.trim[end;/]

    Выведет…
    /section/subsection




    Пример отсечения white space

    $name[Вася ]
    "$name"
    "^name.trim[]"

    Выведет…
    " Вася "
    "Вася"




    Пример вертикального разбиения

    $str[О, сколько нам открытий чудных!…]
    $parts[^str.split[
    нам]]
    ^parts.save[parts.txt]

    Создает на диске файл parts.txt, содержащий следующее:
    piece
    О, сколько
    открытий чудных!…





    Примеры использования

    $str[www.parser.ru?user=admin]
    ^if(^str.match[
    \? #есть разделитель
    .+ #и есть хоть что-то за ним
    ][x]){
    Есть совпадение}{Совпадений нет}

    Выведет на экран: Есть совпадение.

    Внимание: настоятельно советуем задавать комментарии к частям сложного регулярного выражения. Бывает, что даже вам самим через какое-то время бывает трудно в них разобраться. Для этого включите опцию x, разрешающую расширенный синтаксис выражений, допускающий комментарии.

    $str[www.parser.ru?user=admin]
    $mtc[^str.match[(\?.+)][']]
    ^mtc.save[match.txt]

    Создаст файл match.txt, содержащий такую таблицу:

    prematch
    match
    postmatch
    1
    www.parser.ru
    ?user=admin
    ?user=admin





    Примеры использования

    $str[123]
    ^str.int[]

    Выведет число 123, поскольку объект str можно преобразовать к классу int.

    $str[много]
    ^str.double(-1)


    Выведет число -1, поскольку преобразование невозможно.




    Принятые обозначения

    ABCDEFGH - Код Parser в примерах для визуального отличия от HTML (Courier New, 10). Для удобства работы с электронной документацией дополнительно выделен цветом.

    ABCDEFGH - Файлы и каталоги, рассматриваемые в рамках урока.

    ABCDEFGH - Дополнительная и справочная информация.

    [3.1] - Номер версии Parser, начиная с которой доступна данная функция или опция.

    В справочнике символ "|" равнозначен союзу ИЛИ.




    Рекомендуемый способ анализа

    Временно добавьте в конец вашего скрипта вызов…
    ^rusage[total]

    …вот этого метода…

    @rusage[comment][v;now;prefix;message;line;usec]
    $v[$status:rusage]
    $now[^date::now[]]
    $usec(^v.tv_usec.double[])
    $prefix[[^now.sql-string[].^usec.format[%06.0f]] $env:REMOTE_ADDR: $comment]
    $message[$v.utime $v.stime $request:uri]
    $line[$prefix $message ^#0A]
    ^line.save[append;/rusage.log]
    $result[]

    …и проанализируйте журнал.

    Для более точного анализа, добавьте вызовы…
    ^rusage[before XXX]
    ^rusage[after XXX]

    …вокруг интересующего вас блока.

    Примечание: для записи журнала не рекомендуется использовать веб-пространство.


    Рекомендуемый способ анализа

    Временно добавьте вызовы…
    ^musage[before XXX]
    ^musage[after XXX]

    …вокруг интересующего вас блока вот этого метода…

    @musage[comment][v;now;prefix;message;line]
    $v[$status:memory]
    $now[^date::now[]]
    $prefix[[^now.sql-string[]] $env:REMOTE_ADDR: $comment]
    $message[$v.used $v.free $v.ever_allocated_since_compact $v.ever_allocated_since_start $request:uri]
    $line[$prefix $message ^#0A]
    ^line.save[append;/musage.log]
    $result[]

    …и проанализируйте журнал.

    Важно: в ходе работы Parser захватывает у операционной системы дополнительные блоки памяти по мере необходимости. Поэтому есть моменты, когда и used и free, увеличиваются. Это нормально.

    Примечание: для записи журнала не рекомендуется использовать веб-пространство.




    Replace. Замена подстрок в строке

    ^строка.replace[$таблица_подстановок]

    Эффективно заменяет подстроки в строке в соответствии с таблицей подстановок, работает существенно быстрее match.

    Таблица подстановок - объект класса table, содержащая два столбца:
    первый - подстрока, которую нужно заменить,
    второй - подстрока, которая появится на месте подстроки из первого столбца после замены.

    Имена столбцов несущественны, можно называть их from/to, или вообще никак не называть, воспользовавшись nameless таблицей.


    Rusage. Информация о затраченных ресурсах

    Это поле - хеш с информацией о ресурсах сервера, затраченных на данный момент системой на обработку вашего Parser-скрипта.
    Не все операционные системы умеют возвращать эти значения (WinNT/Win2000/WinXP умеет все, Win98 умеет только tv_sec и tv_usec [3.0.8]).

    Ключ
    Единица
    Описание значения
    Как уменьшить?
    utime
    секунда
    Чистое время, затраченное текущим процессом
    (не включает время, когда работали другие задачи)
    Упростить манипуляции с данными внутри Parser (улучшить алгоритм, переложить часть действий на SQL-сервер)
    stime
    секунда
    Время, сколько система читала ваши файлы, каталоги, библиотеки
    Уменьшить количество и размер необходимых для работы файлов, не подключать ненужные для обработки данного документа модули
    maxrss
    блок
    Память, занимаемая процессом
    Уменьшить количество загружаемых ненужных данных.
    Найти и исправить все «select* …», задав список действительно необходимых полей. Не загружать из SQL-сервера ненужные записи, отфильтровать как можно больше средствами самого SQL-сервера.
    Точное системное время. Позволяет оценить траты времени на ожидание ответа от SQL-, HTTP-, SMTP-серверов.

    Сколько прошло с Epoch…
    Упростить SQL запросы, для MySQL воспользуйтесь EXPLAIN, см. http://www.mysql.com/doc/en/EXPLAIN.html; для Oracle: EXPLAIN PLAN, см. документацию по серверу; для других SQL-серверов: см. их документацию.
    tv_sec
    секунда
    …целых секунд;
    tv_usec
    микросекунда
    (10E-6)
    …еще прошло микросекунд
    (миллионных долей секунды)




    Save. Сохранение строки в файл

    ^строка.save[имя_файла_с_путем]
    ^строка.save[append;имя_файла_с_путем]

    Сохраняет или добавляет строку в файл по указанному пути.

    При этом с фрагментами строки производятся необходимые преобразования, см. «Преобразование данных».


    Split. Разбиение строки

    ^строка.split[разделитель]
    ^строка.split[разделитель;опцииразбиения]

    Разбивает строку на подстроки относительно подстроки-разделителя и формирует объект класса table, содержащий
    ·либо таблицу со столбцом piece, в который помещаются части исходной строки,
    ·либо безымянную таблицу с частями исходной строки в колонках единственной записи.

    Предусмотрены следующие опции разбиения:
    l - разбить слева направо (по-умолчанию);
    r - разбить справа налево;
    h - сформировать безымянную таблицу где части исходной строки помещаются горизонтально;
    v - сформировать таблицу со столбцом piece, где части исходной строки помещаются вертикально.


    Sql. Получение строки из базы данных

    ^string:sql{SQL-запрос}
    ^string:sql{SQL-запрос}[$.limit(1) $.offset(o) $.default{код}]


    Возвращает строку, полученную из базы данных через SQL-запрос. Результатом выборки должен быть только один столбец из одной строки. Для работы оператора необходимо установленное соединение с сервером базы данных (см. оператор connect).

    Необязательные параметры:
    $.limit(1) - в ответе заведомо будет содержаться только одна строка
    $.offset(o) - отбросить первые o записей выборки
    $.default{код} - если ответ SQL-сервера был пуст (0 записей), то будет выполнен указанный код, и строка, которую он возвратит, будет результатом метода


    Строковые литералы

    В коде Parser могут использоваться любые буквы, включая русские. Следующие символы являются служебными:

    ^ $ ; (
    ) [ ] {
    } " :

    Чтобы отменить специальное действие этих символов, их необходимо предварять символом ^. Например, для получения в тексте символа $ нужно записать ^$.

    Кроме того, допустимо использовать код символа:
    ^#20 - пробел
    ^#XX - XX hex код буквы




    Tid. Идентификатор потока

    Идентификатор потока (thread) операционной системы, в котором работает Parser.




    Trim. Отсечение букв с концов строки[3.1.2]

    ^строка.trim[]
    ^строка.trim[откуда]
    ^строка.trim[откуда;буквы]

    Метод отсекает буквы с концов строки. По умолчанию отсекаются white space символы с начала и конца строки.
    Можно указать, откуда именно отсекать символы, задав одно из значений:
    ·start - отсекать с начала;
    ·both - отсекать и с начала и с конца;
    ·end - отсекать с конца.

    Также можно указать те буквы, которые необходимо отсечь.


    Upper, lower. Преобразование регистра строки

    ^строка.upper[]
    ^строка.lower[]

    Переводят строку в верхний или нижний регистр соответственно. Для их работы необходимо, чтобы был задан $request:charset.


    Вызов статического метода

    ^класс:метод[параметры]

    Вызов статического метода класса.

    Примечание: точно так же вызываются динамические методы родительского класса (см. Создание пользовательского класса).


    WinNT/2K/XP

    Под этими OS доступен ряд дополнительных значений:

    Ключ
    Единица
    Описание значения
    Как уменьшить?
    ReadOperationCount
    ReadTransferCount

    штук
    байт
    Количество операций чтения с диска и суммарное количество считанных байт
    Уменьшить количество и размер необходимых для работы файлов, не подключать ненужные для обработки данного документа модули.

    Больше использовать SQL-сервер, меньше файлы.
    WriteOperationCount
    WriteTransferCount

    штука
    байт
    Количество операций записи на диск и суммарное количество записанных байт

    OtherOperationCount
    OtherTransferCount

    штука
    байт
    Количество других операций с диском (не чтения/записи) и суммарное количество переданных байт

    PeakPagefileUsage
    QuotaPeakNonPagedPoolUsage
    QuotaPeakPagedPoolUsage

    байт
    Максимальное количество памяти в файле подкачки (swap-файле)
    см. комментарий к maxrss выше.




    Задание статического поля

    $класс:поле[значение]

    Задание значения статического поля класса.




    Значение статического поля

    $класс:поле

    Получение значения статического поля класса.


    А что это вообще такое?

    Append. Добавление данных в таблицу

    ^таблица.append{табличныеданные}

    Метод добавляет запись в конец таблицы. Формат представления данных - tab-delimited.
    Табличные данные должны иметь такую же структуру, как и таблица, в которую добавляются данные.


    Columns. Получение структуры таблицы.

    ^таблица.columns[]

    Метод создает именованную таблицу из одного столбца column, содержащего заголовки столбцов исходной таблицы.


    Count. Количество строк в таблице

    ^таблица.count[]

    Выдает количество строк в таблице (int).


    Create. Копирование существующей таблицы

    ^table::create[таблица]
    ^table::create[таблица;опции]

    Конструктор создает объект класса table, копируя данные из другой таблицы. Также можно задать ряд опций, контролирующих копирование, см. «Опции копирования».





    Create. Создание объекта на основе заданной таблицы

    ^table::create{табличные_данные}
    ^table::create[nameless]{табличные_данные}

    Конструктор создает объект класса table, используя табличные данные, определенные в самом конструкторе.

    Табличные данные - данные, представленные в формате tab-delimited, то есть столбцы разделяются символом табуляции, а строки - символом перевода строки. При этом части первой строки, разделенные символом табуляции, рассматриваются как имена столбцов, и создается именованная таблица. Пустые строки игнорируются. Если необходимо получить таблицу без имен столбцов (что не рекомендуется), то перед заданием табличных данных необходимо указать параметр nameless. В этом случае столбцы первой строки воспринимаются конструктором как данные таблицы, а в качестве имен столбцов выступят их порядковые номера, начиная с нулевого.


    Flip. Транспонирование таблицы

    ^таблица.flip[]

    Создает новую nameless таблицу с записями, полученными в результате транспонирования исходной таблицы. Иными словами, метод превращает столбцы исходной таблицы в строки, а строки в столбцы.


    Hash. Преобразование таблицы к хешу с заданными ключами

    ^таблица.hash[ключ]
    ^таблица.hash[ключ][опции]
    ^таблица.hash[ключ][столбец значений]
    ^таблица.hash[ключ][столбец значений][опции]
    ^таблица.hash[ключ][таблица со столбцами значений]
    ^таблица.hash[ключ][таблица со столбцами значений][опции]

    Ключ может быть задан, как:
    ·[строка] - название столбца, значение которого считается ключом;
    ·{код} - результат исполнения которого считается ключом;
    ·(математическое выражение) - результат вычисления которого считается ключом.

    Метод преобразует таблицу к хешу вида:
    $хеш[
    $.значение_ключа[
    $.название_столбца[значение_столбца]

    ]

    ]

    Иными словами, метод создает хеш, в котором ключами являются значения, описанные параметром ключ. При этом каждому ключу ставится в соответствие хеш, в котором для всех столбцов таблицы хранятся ассоциации «название столбца - значение столбца в записи».

    Если задан столбец значений, то каждому ключу будет соответствовать хеш с одной ассоциацией «название столбца - значение столбца в записи».

    Кроме того, можно задать несколько столбцов значений, для этого необходимо передать дополнительным параметром таблицу, в которой перечислены все необходимые столбцы.

    Опции - хеш с опциями преобразования.
    $.distinct(0/1)
    0=наличие в ключевом столбце одинаковых значений считается ошибкой (по-умолчанию);
    1=выбрать из таблицы записи с уникальным ключом.
    $.distinct[tables]
    создать хеш из таблиц, содержащих строки с ключом. [3.0.8]




    Join. Объединение двух таблиц

    ^таблица1.join[таблица2]
    ^таблица1.join[таблица2;опции]

    Метод добавляет в конец таблицы1 записи из таблицы2. При этом из таблицы2 будет взято значение из столбца, одноименного столбцу таблицы1, или пустая строка, если такой столбец не найден.
    Также можно задать ряд опций, контролирующих добавление, см. «Опции копирования».


    Класс table

    Класс предназначен для работы с таблицами.

    Таблица считается определенной (def), если она не пуста. Числовое значение равно количеству строк таблицы.




    Load. Загрузка таблицы с диска или HTTP-сервера

    ^table::load[имя файла]
    ^table::load[имя файла;опции загрузки]
    ^table::load[nameless;имя файла]

    ^table::load[nameless;имя файла;опции загрузки]

    Конструктор создает объект, используя таблицу, определенную в некотором файле или документе на HTTP-сервере. Данные должны быть представлены в формате tab-delimited (см. table::create).

    Имя файла - имя файла с путем или URL документа на HTTP-сервере.

    Опции загрузки - об основных опция см. раздел «Работа с HTTP-серверами», также доступны дополнительные опции, см. «Опции формата файла».

    Использование параметра nameless такое же, как и в конструкторе table::create.


    Locate. Поиск в таблице

    ^таблица.locate[столбец;искомое_значение]
    ^таблица.locate(логическое_выражение)
    ^таблица.locate[столбец;искомое_значение;опции]

    ^таблица.locate(логическое_выражение)[опции]

    Метод ищет в указанном столбце значение, равное искомому и возвращает логическое значение «истина/ложь» в зависимости от успеха поиска. В случае если искомое значение найдено, строка, его содержащая, делается текущей. Если искомое значение найдено не было, указатель текущей строки не меняется.
    Второй вариант вызова метода ищет первую запись, для которой истинно логическоевыражение.

    Также можно задать ряд опций, контролирующих поиск, см. «Опции поиска».

    Поиске чувствителен к регистру букв.


    Menu. Последовательный перебор всех строк таблицы

    ^таблица.menu{код}
    ^таблица.menu{код}[разделитель]
    ^таблица.menu{код}{разделитель}


    Метод menu выполняет код для каждой строки таблицы, последовательно перебирая все строки.

    Разделитель - код, который вставляется после каждой непустой строки, кроме последней. Разделитель в квадратных скобках вычисляется один раз, в фигурных - много раз по ходу вызова.


    Offset и line. Получение смещения указателя текущей строки

    ^таблица.offset[]

    Метод offset без параметров возвращает текущее смещение указателя текущей строки от начала таблицы.


    Offset. Смещение указателя текущей строки

    ^таблица.offset(число)
    ^таблица.offset[cur|set](число)

    Смещает указатель текущей строки на указанное число вниз. Если аргумент метода отрицательный, то указатель перемещается вверх. Смещение указателя осуществляется циклически, то есть, достигнув последней строки таблицы, указатель возвращается на первую.

    Необязательный параметр:
    cur - смещает указатель относительно текущей строки
    set - смещает указатель относительно первой строки


    Опции формата файла[3.1.2]

    При загрузке и записи файла можно задать символы-разделители столбцов и символы, обрамляющие значения столбцов.

    Опция
    По-умолчанию
    Описание
    $.separator[символ]
    табуляция
    Задает символ, разделитель столбцов
    $.encloser[символ]
    нет
    Задает символ, обрамляющий значение столбца.




    Опции копирования и поиска

    При копированнии записей из одной таблицы в другую, см…
    table::create [3.0.7]
    table.join
    [3.0.7]
    и при поиске, см…
    table.locate [3.0.8]

    можно задать хеш опций:
    $.offset(количество строк)
    пропустить указанное количество строк таблицы;
    $.offset[cur]
    с текущей строки таблицы;
    $.limit(максимум)
    максимум строк, которые можно обработать;
    $.reverse(1/0)
    1=в обратном порядке. [3.0.8]





    Получение содержимого столбца

    $таблица.поле
    Возвращает содержимое столбца поле из текущей строки таблицы.


    Получение содержимого текущей строки в виде хеша

    $таблица.fields - содержимое текущей строки таблицы в виде хеша.

    Возвращает содержимое текущей строки таблицы в виде хеша. При этом имена столбцов становятся ключами хеша, а значения столбцов - соответствующими значениями ключей.

    Использовать этот метод необходимо, если имена столбцов совпадают с именами методов или конструкторов класса table. В таком случае получить их значения напрямую нельзя - Parser будет выдавать сообщение об ошибке. Если необходимо работать с полями, называющимися именно так, можно воспользоваться полем fields, и далее работать уже не с таблицей, а с хешем.


    В результате выполнения кода

    $emergency[^table::create{id number
    fire 01
    police 02
    ambulance 03
    gas 04
    }]
    $fliped[^emergency.flip[]]
    ^fliped.save[fliped.txt]

    В результате выполнения кода в файл fliped.txt будет сохранена такая таблица:

    0

    1

    2

    3

    fire

    police

    ambulance

    gas

    01

    02

    03

    04

    В результате будут выведены значения

    $tab[^table::create{menu line
    yes first
    no second}
    ]
    $tab_hash[$tab.fields]
    $tab_hash.menu
    $tab_hash.line

    В результате будут выведены значения полей menu и line (имена которых совпадают с именами методов класса table) как значения ключей хеша tab_hash.

    Все записи таблицы

    ^stuff.join[$just_hired_people]

    Все записи таблицы $just_hired_people будут добавлены в таблицу $stuff.

    Людям более привычно считать записи,

    $men[^table::create{name
    Вася
    Петя
    Сережа
    }]
    ^men.menu{
    ^men.offset[] - $men.name
    }[
    ]

    Выдаст:
    0 - Вася
    1 - Петя
    2 - Сережа

    Людям более привычно считать записи, начиная с единицы. Для удобного вывода нумерованных списков имеется метод line:

    ^таблица.line[]

    Он позволяет сразу получить номер записи из таблицы в привычном виде, когда номер первой строки равен единице. Если в примере использовать ^men.line[], то нумерация будет идти от одного до трех.

    и будет найдена первая запись

    $stuff[^table::create{name pos status
    Александр босс 1
    Сергей технолог 1
    Тема арт-директор 2
    }]
    ^if(^stuff.locate[name;Тема]){
    Запись найдена в строке номер ^stuff.line[].
    $stuff.name: $stuff.pos

    }{
    Запись не найдена
    }

    На экран будет выведено:
    Запись найдена в строке номер 3.
    Тема: арт-директор

    Подставьте такой поиск в пример…
    ^stuff.locate($stuff.status>1)

    … и будет найдена первая запись со статусом, большим 1.

    Пример выводит все содержимое таблицы

    $goods[^table::create{ pos good price
    1 Монитор 1000
    2 Системный блок 1500
    3 Клавиатура 15}
    ]

    ^goods.menu{





    }
    $goods.pos $goods.good $goods.price


    Пример выводит все содержимое таблицы $goods в виде HTML-таблицы.

    содержащая последнюю строку таблицы из


    ^goods.offset(-1)





    $goods.pos $goods.good $goods.price


    Результатом выполнения кода будет HTML-таблица, содержащая последнюю строку таблицы из предыдущего примера (метод menu).

    conf будет сохранена

    ^conf.save[/conf/old_conf.txt]

    Таблица $ conf будет сохранена в текстовом файле old_conf.txt в каталоге /conf/.

    thoseAbove20 попадут строки

    $men[^table::create{name age
    Serge 26
    Alex 20
    Mishka 29
    }]
    $thoseAbove20[^men.select($men.age>20)]

    В $ thoseAbove20 попадут строки с Serge и Mishka.

    men будут отсортированы по столбцу

    $men[^table::create{name age
    Serge 26
    Alex 20
    Mishka 29
    }]
    ^men.sort{$men.name}
    ^men.menu{
    $men.name: $men.age
    }[
    ]

    В результате записи таблицы $ men будут отсортированы по столбцу name (по строке имени):
    Alex: 20
    Mishka: 29
    Serge: 26

    А можно отсортировать по столбцу age (по числу прожитых лет) по убыванию (desc), измените в примере вызов sort на такой…
    ^men.sort($men.age)[desc]

    …получится…
    Mishka: 29
    Serge: 26
    Alex: 20

    В результате будет создан объект,

    $sql_table[^table::sql{select * from news}]

    В результате будет создан объект, содержащий все записи из таблицы news.

    Примечание: всегда указывайте конкретный список необходимых вам полей.
    Использование «*» крайне не рекомендуется, поскольку постороннему читателю (или вам самим через некоторое время) непонятно, что же за поля будут извлечены. Кроме того, так можно извлечь лишние поля (скажем, добавившиеся в ходе развития проекта), что повлечет ненужные расходы на их извлечение и хранение.

    в котором каждый товар имеет

    Есть список товаров, в котором каждый товар имеет наименование и уникальный код - id. Есть прайс-лист товаров, имеющихся в наличии. Вместо названия товара используется id товара из списка товаров. Все это хранится в двух таблицах. Подобные таблицы называются связанными. Нам нужно получить данные в виде «товар - цена», т.е. получить данные сразу из двух таблиц.

    Реализация:
    # это таблица с нашими товарами
    $product_list[^table::create{id name
    1 хлеб
    2 колбаса
    3 масло
    4 водка
    }]

    # это таблица с ценами товаров
    $price_list[^table::create{id price
    1 6.50
    2 70.00
    3 60.85
    }]

    #hash таблицы с ценами по полю id
    $price_list_hash[^price_list.hash[id]]

    #перебираем записи таблицы с товарами
    ^product_list.menu{
    $product_price[$price_list_hash.[$product_list.id].price]
    # проверяем - есть ли цена на товар в нашем hash
    ^if($product_price){
    # печатаем название товара и его цену
    $product_list.name - $product_price

    }{
    # а у этого товара нет цены, т.е. его нет в наличии
    $product_list.name - нет в наличии

    }
    }

    В результате получим:
    хлеб - 6.50
    колбаса - 70.00
    масло - 60.85
    водка - нет в наличии

    stuff новую запись

    $stuff[^table::create{name pos
    Alexander boss
    Sergey coder
    }]
    ^stuff.append{Nikolay designer}
    ^stuff.save[stuff.txt]

    Пример добавит в таблицу $ stuff новую запись и сохранит таблицу в файл stuff.txt.

    $columns_table

    $columns_table[^stuff.columns[]]

    В выражениях числовое значение таблицы

    $goods[^table::create{ pos good price
    1 Монитор 1000
    2 Системный блок 1500
    3 Клавиатура 15}
    ]
    Количество: ^goods.count[]

    Выведет:
    Количество: 3

    В выражениях числовое значение таблицы равно количеству строк:
    ^if($goods > 2){больше}

    Будет создан объект tab класса

    $tab[^table::create{name age
    Вова 27
    Леша 22}
    ]

    Будет создан объект tab класса table, содержащий таблицу из двух строк с именами столбцов name и age.

    $orig

    $orig[^table::create{name
    Вася
    Коля
    Маша
    }]

    #сдвигает текущую запись таблицы orig на «Коля»
    ^orig.offset(1)

    #копирует, начиная с текущей записи в orig, не больше 10 записей
    $copy[^table::create[$orig;
    $.offset[cur]
    $.limit(10)
    ]]

    ^copy.menu{$copy.name}[,]

    Выведет…
    Коля, Маша

    в колонке name текущей строки

    $tab.name

    Пример вернет значение, определенное в колонке name текущей строки таблицы.

    Пример загрузки таблицы с диска

    $loaded_table[^table::load[/addresses.cfg]]

    Пример создает объект класса table, содержащий именованную таблицу, определенную в файле addresses.cfg, который находится в корневом каталоге веб-сайта.


    Пример загрузки таблицы с HTTP-сервера

    $table[^table::load[nameless;http://www.parser.ru/;
    $.
    USER-AGENT[table load example]
    ]]
    Количество строк: ^table.count[]


    $table.0





    Пример загрузки .txt файла, созданного Miscrosoft Excel

    Excel умеет сохранять данные в простой текстовый файл, разделенный табуляциями:
    Файл|Сохранить как… Текст (Разделенный табуляциями) (.txt).
    Данные сохраняются в следующем формате:
    name
    description
    "ООО ""Петров и партнеры"""
    Текст

    (Значения ряда столбцов обрамляется кавычками, которые внутри самого значения удваиваются)

    Чтобы считать такой файл, необходимо указать соответствующую опцию загрузки:
    $companies[^table::load[companies.txt;
    $.encloser["]
    ]]
    $companies.name

    Parser также может работать и с .csv файлами, достаточно указать опцию:
    $.separator[^;]




    Save. Сохранение таблицы в файл

    ^таблица.save[путь]
    ^таблица.save[путь;опции][3.1.2]
    ^таблица.save[nameless;путь]
    ^таблица.save[nameless;путь;опции]
    [3.1.2]

    Сохраняет таблицу в текстовый файл в формате tab-delimited. Использование опции nameless сохраняет таблицу без имен столбцов.
    Также доступны опции записи, см. «Опции формата файла», позволяющие, например, сохранить файл в .csv формате, для последующей загрузки данных в программы, которые понимают такой формат (Miscrosoft Excel).


    Select. Отбор записей

    ^таблица.select(критерий_отбора)

    Метод последовательно перебирает все строки таблицы, применяя к ним выражение критерий_отбора, те строки, которые подпали под заданный критерий (логическое выражение было истинно), помещаются в результат, которым является таблица с такой же структурой, что и входная.


    Sort. Сортировка данных таблицы

    ^таблица.sort{функциясортировки_по_строке}
    ^таблица.sort{функция_сортировки_по_строке}[направление_сортировки]
    ^таблица.sort(функция сортировки_по_числу)
    ^таблица.sort(функция_сортировки_по_числу)[направление_сортировки]

    Метод осуществляет сортировку таблицы по указанной функции.

    Функция сортировки - произвольная функция, по текущему значению которой принимается решение о положении строки в отсортированной таблице. Значением функции может быть строка (значения сравниваются в лексикографическом порядке) или число (значения сравниваются как действительные числа).

    Направление сортировки - параметр, задающий направление сортировки. Может быть:
    desc - по убыванию
    asc - по возрастанию
    По умолчанию используется сортировка по возрастанию.


    Sql. Выборка таблицы из базы данных

    ^table::sql{SQL-запрос}
    ^table::sql{SQL-запрос }[$.limit(n) $.offset(o)]

    Конструктор создает объект класса table, содержащий таблицу, полученную в результате выборки из базы данных.
    Для использования конструктора необходимо установленное соединение с сервером базы данных (см. оператор connect).

    SQL-запрос - запрос на выборку из базы данных

    Возможно использование дополнительных параметров конструктора:
    $.limit(n) - получить не более n записей
    $.offset(o) - отбросить первые O записей выборки


    Внешние и внутренние данные

    Создавая код на Parser, мы имеем дело с двумя видами данных. Один из них - это все то, что написано самим кодером. Второй - данные, получаемые кодом извне, а именно из форм, переменных окружения, файлов и от SQL-серверов. Все то, что создано кодером, не нуждается в проверке на корректность. Вместе с тем, когда данные поступают, например, от пользователей через поля форм, выводить их «as-is» (как есть) может быть потенциально опасно. Возникает необходимость преобразования таких данных по определенным правилам. Большую часть работы по подобным преобразованиям Parser выполняет автоматически, не требуя вмешательства со стороны. Например, если присутствует вывод данных, введенных через поле формы, то в них символы < > автоматически будут заменены на < и >. Иногда наоборот бывает необходимо позволить вывод таких данных именно в том виде, в котором они поступили.

    Для Parser «свой» код, т.е. тот, который набрал кодер, считается clean («чистым»). Все данные, которые поступают извне, считаются tainted («грязными» или «окрашенными»).

    код Parser - этот код создан скриптовальщиком, поэтому никаких вопросов не вызывает;

    $form:field - здесь должны быть выведены данные, введенные пользователем через форму;

    $my_table[^table::sql{запрос}] - здесь данные поступают из БД.

    В случае с $form:field, поступившие tainted данные будут автоматически преобразованы и некоторые символы заменятся в соответствии с внутренней таблицей замен Parser. После этого они станут clean («чистыми»), и их «окрашенность» исчезнет. Здесь неявно выполняется операция untaint (снять «окраску»). Автоматическое преобразование данных происходит в тот момент, когда эти данные будут выводиться. Так, в случае с помещением данных, поступивших из БД, в переменную $my_table, преобразование выполнится в тот момент, когда данные будут в каком-либо виде выданы во внешнюю среду (переданы браузеру, сохранены в файл или базу данных).

    Вместе с тем, бывают ситуации, когда необходимости в таком преобразовании нет, либо данные нужно преобразовать по другим правилам, чем это делает Parser по умолчанию. Например, нам нужно разрешить пользователю вводить HTML-теги через поле формы для дополнительного форматирования текста. Но, так как это чревато неприятностями (ввод Java-скрипта в гостевой книге может перенаправлять пользователей с вашего сайта на вражеский), Parser сам заменит «нежелательные» символы в соответствии со своими правилами. Решение - использование оператора untaint.




    А что это вообще такое?

    Int, double

    ^объект.int[]
    ^объект.int(значение по умолчанию)
    ^объект.double[]
    ^объект.double(значение по умолчанию)


    В случае отсутствия объекта эти методы выдают либо 0, либо значение по умолчанию. Эти методы работают, когда выполняется преобразование к int/double не определенных заранее объектов, скажем, полей форм (см. класс form).


    Класс void

    Класс предназначен для работы с «пустыми» объектами. Он не имеет конструкторов, объекты этого класса создаются автоматически, например, когда вы обращаетесь к несуществующей переменной.




    Left, right, mid. Получение подстроки[3.1.2]

    ^строка.left(N)
    ^строка.right(N)
    ^строка.mid(P;N)
    ^строка.mid(P)

    В случае отсутствия объекта эти методы выдают пустую подстроку.


    Length. Длина «строки»

    ^объект.length[]

    В случае отсутствия объекта этот метод выдает 0.


    Переменные

    Переменные могут хранить данные следующих типов:
    ·строка (string);
    ·число (int/double);
    ·истина/ложь;
    ·хеш (ассоциативный массив);
    ·класс объектов;
    ·объект класса (в т.ч. пользовательского);
    ·код;
    ·выражение.

    Для использования переменных не требуется их заранее объявлять.

    В зависимости от того, что будет содержать переменная, для присвоения ей значения используются различные типы скобок:

    $имя_переменной[строка]
    переменной присваивается строковое значение (объект класса string) или произвольный объект некоторого класса

    $имя_переменной(выражение)
    переменной присваивается число или результат математического выражения

    $имя_переменной{код}
    переменной присваивается фрагмент кода, который будет выполнен при обращении к переменной

    Для получения значения переменных используется обращение к имени переменной:

    $имя_переменной - получение значения переменной


    Pos. Получение позиции подстроки

    ^объект.pos[подстрока]

    В случае отсутствия объекта этот метод выдает -1.


    Если поле number определено, то

    ^form:number.int[]

    Если поле number определено, то есть было передано, его значение просто преобразуется к классу int. Если же это поле не определено, то есть его просто нет, несуществующее значение, относящееся к классу void, приравняется к 0, и ничего страшного не произойдет - код будет выполняться дальше.

    Если поле password определено, то

    ^if(^form:password.length[]<$MIN_PASSWORD_LENGTH){
    Длина введенного пароля меньше $MIN_PASSWORD_LENGTH
    }

    Если поле password определено, то есть было передано, вычислится его длина, и будет проверена. Это обычный вызов метода ^строка.length[]. Если же это поле не определено, то есть его просто нет, длина несуществующего значения, относящегося к классу void, считается равным 0, и ничего страшного не произойдет - эта длина будет успешно проверена.

    Если поле email определено, то

    ^form:email.left(50)

    Если поле email определено, то есть было передано - это обычный вызов метода ^строка.left[]. Если же это поле не определено, то есть его просто нет, считается что в несуществующем значении, относящемся к классу void, никакие подстроки не существуют, и ничего страшного не произойдет - будет успешно выдана пустая строка.

    Если поле email определено, то

    ^if(^form:email.pos[@]>0){
    Может быть…
    }

    Если поле email определено, то есть было передано - это обычный вызов метода ^строка.pos[]. Если же это поле не определено, то есть его просто нет, считается что в несуществующем значении, относящемся к классу void, никакие подстроки не существуют, и ничего страшного не произойдет - будет успешно выполнена проверка.

    В результате выполнения этого кода

    ^connect[строка подключения]{
    ^void:sql{create table users(id int,name text,email text)}
    }

    В результате выполнения этого кода в базе данных будет создана таблица users, при этом запрос не вернет никакого результата. Пример дан для СУБД MySQL.

    В качестве части имени может

    Код
    Результат
    $string[2+2]
    $string

    2+2
    $number(2*2)
    $number

    4

    $i(0)
    $code{$i}
    $i(1)
    $code

    1
    $i(0)
    $string[$i]
    $i(1)
    $string
    0


    В качестве части имени может быть использовано…

    …значение другой переменной:
    $superman[value of superman variable]
    $part[man]
    $super$part
    Возвратит: value of superman variable

    $name[picture]
    ${name}.gif
    Возвратит строку picture.gif, а не значение поля gif объекта picture.

    …результат работы кода:
    $field.[b^eval(2+3)]
    Возвратит значение поля b5 объекта field.

    Sql. Запрос к БД, не возвращающий результат

    ^void:sql{SQL-запрос}

    Осуществляет выполнение SQL-запроса, который не возвращает результат (операции по управлению данными в базе данных).
    Для работы этого метода необходимо установленное соединение с сервером базы данных (см. оператор connect).


    А что это вообще такое?

    Create. Создание документа на основе заданного XML

    ^xdoc::create{XML-код}
    ^xdoc::create[базовый_путь]{XML-код}

    Конструктор создает объект класса xdoc из XML-кода. Возможно задание базовогопути.





    Create. Создание нового пустого документа

    ^xdoc::create[имя_тега]
    ^xdoc::create[базовый_путь;имя_тега]

    Конструктор создает объект класса xdoc, состоящий из единственного тега имя_тега. Возможно задание Базовогопути.





    DOM

    DOM1-интерфейс Document:

    $Element[^документ.createElement[tagName]]
    $DocumentFragment[^документ.createDocumentFragment[]]
    $Text[^документ.createTextNode[data]]
    $Comment[^документ.createComment[data]]
    $CDATASection[^документ.createCDATASection[data]]
    $ProcessingInstruction[^документ.createProcessingInstruction[target;data]]
    $Attr[^документ.createAttribute[name]]
    $EntityReference[^документ.createEntityReference[name]]
    $NodeList[^документ.getElementsByTagName[tagname]]

    DOM2-интерфейс Document:

    $Node[^документ.importNode[importedNode](deep)]
    $Element[^документ.createElementNS[namespaceURI;qualifiedName]] [3.1.1]
    $Attr[^документ.createAttributeNS[namespaceURI;qualifiedName]] [3.1.1]
    $NodeList[^документ.getElementsByTagNameNS[namespaceURI;localName]]
    $Element[^документ.getElementById[elementId]]

    В Parser
    ·DOM-интерфейсы Node и Element и их производные реализованы в классе xnode;
    ·DOM-интерфейс NodeList - класс hash с ключами 0, 1, …;
    ·DOM-тип DOMString - класс string;
    ·DOM-тип boolean - логическое значение: 0=ложь, 1=истина.

    Подробная спецификация DOM1 доступна здесь: http://www.w3.org/TR/1998/REC-DOM-Level-1-19981001/level-one-core.html

    Подробная спецификация DOM2 доступна здесь: http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/core.html




    DOM

    DOM1-интерфейс Node:

    $узел.nodeName
    $узел.nodeValue
    $узел.nodeValue[новое значение]
    [3.1.2]
    ^if($узел.nodeType == $xnode:ELEMENT_NODE){…}
    $Node[$узел.parentNode]
    $NodeList[$узел.childNodes]
    $Node[$узел.firstChild]
    $Node[$узел.lastChild]
    $Node[$узел.previousSibling]
    $Node[$узел.nextSibling]
    $NodeList[$узел_типа_ELEMENT.attributes]
    $Document[$node.ownerDocument]

    DOM1-интерфейс Element:

    $узел_типа_ELEMENT.tagName

    DOM1-интерфейс Attr:

    $узел_типа_ATTRIBUTE.name
    ^if($узел_типа_ATTRIBUTE.specified){…}
    $узел_типа_ATTRIBUTE.value

    DOM1-интерфейс ProcessingInstruction:

    $узел_типа_PROCESSING_INSTRUCTION.target
    $узел_типа_PROCESSING_INSTRUCTION.data

    DOM1-интерфейс DocumentType:

    $узел_типа_DOCUMENT_TYPE.name
    $узел_типа_DOCUMENT_TYPE.entities
    $узел_типа_DOCUMENT_TYPE.
    notations

    DOM1-интерфейс Notation:

    $узел_типа_NOTATION.publicId
    $узел_типа_NOTATION.systemId


    В Parser
    ·DOM-интерфейс NodeList - класс hash с ключами 0, 1, …;
    ·DOM-тип DOMString - класс string;
    ·DOM-тип boolean - логическое значение: 0=ложь, 1=истина.

    Подробная спецификация DOM1 доступна здесь: http://www.w3.org/TR/1998/REC-DOM-Level-1-19981001/level-one-core.html

    Подробная спецификация DOM2 доступна здесь: http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/core.html




    DOM

    DOM1-интерфейс Document:

    $DocumentType[$документ.doctype]
    $Element[$документ.documentElement]


    В Parser DOM-интерфейсы Node и Element и их производные реализованы в классе xnode.

    Подробная спецификация DOM1 доступна здесь: http://www.w3.org/TR/1998/REC-DOM-Level-1-19981001/level-one-core.html




    DOM1.

    DOM1-интерфейс Node:

    $Node[^узел.insertBefore[$newChild;$refChild]]
    $Node[^узел.replaceChild[$newChild;$oldChild]]
    $Node[^узел.removeChild[$oldChild]]
    $Node[^узел.appendChild[$newChild]]
    ^if(^узел.hasChildNodes[]){…}
    $Node[^узел.cloneNode(deep)]

    DOM1-интерфейс Element:

    ^узел.getAttribute[name]
    ^узел.setAttribute[name;value]
    ^узел.removeAttribute[name]
    $Attr[^узел.getAttributeNode[name]]
    $Attr[^узел.setAttributeNode[$newAttr]]
    $Attr[^узел.removeAttributeNode[$oldAttr]]
    $NodeList[^узел.getElementsByTagName[name]]
    ^узел.normalize[]

    DOM2-интерфейс Element:

    $строка[^узел.getAttributeNS[namespaceURI;localName]] [3.1.1]
    ^узел.setAttributeNS[namespaceURI;localName;value] [3.1.1]
    ^узел.removeAttributeNS[namespaceURI;localName] [3.1.1]
    $Attr[^узел.getAttributeNodeNS[namespaceURI;localName]] [3.1.1]
    $Attr[^узел.setAttributeNodeNS[$newAttr]] [3.1.1]
    $NodeList[^узел.getElementsByTagNameNS[namespaceURI;localName]]
    ^if(^узел.hasAttribute[name]){…} [3.1.1]
    ^if(^узел.hasAttributeNS[namespaceURI;localName]){…} [3.1.1]


    В Parser
    ·DOM-интерфейс NodeList - класс hash с ключами 0, 1, …;
    ·DOM-тип DOMString - класс string;
    ·DOM-тип boolean - логическое значение: 0=ложь, 1=истина.

    Подробная спецификация DOM1 доступна здесь: http://www.w3.org/TR/1998/REC-DOM-Level-1-19981001/level-one-core.html

    Подробная спецификация DOM2 доступна здесь: http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/core.html




    File. Преобразование документа к объекту класса file

    ^документ.file[Параметры_преобразования_в_текст]

    Преобразует документ к типу file. Возможно задание параметров_преобразования в текст.
    По умолчанию создается XML-представление документа с заголовком (можно отключить вывод заголовка, задав соответствующий параметр).


    Класс xdoc

    Класс предназначен для работы с древовидными структурами данных в паре с xnode, и поддерживает считывание файлов в XML формате и запись в XML (http://www.w3.org/XML) и HTML, а также XSLT (http://www.w3.org/TR/xslt) трансформацию.

    Работа с деревом производится в DOM-модели (http://www.w3.org/DOM), доступен DOM1 и ряд возможностей DOM2.

    Класс реализует DOM-интерфейс Document и является наследником класса xnode.

    Ошибки DOM-операций (интерфейс DOMException) преобразуются в исключения xml-типа.




    Класс xnode

    Класс предназначен для работы с древовидными структурами данных в паре с xdoc, поддерживает XPath (http://www.w3.org/TR/xpath) запросы.

    Класс реализует DOM-интерфейсы Node и Element и их производные.
    Класс напрямую не создается, используются соответствующие методы класса xdoc.

    Вместо DOM-интерфейса NamedNodeMap в Parser используется класс hash.




    Load. Загрузка XML с диска, HTTP-сервера или иного источника

    ^xdoc::load[имя файла]

    Конструктор загружает XML-код из некоторого файла или адреса на HTTP-сервере и создает на его основе объект класса xdoc.
    Parser может считать XML из произвольного источника, см. раздел «Чтение XML из произвольного источника».

    имяфайла - имя файла с путем или URL файла на HTTP-сервере.


    Параметр создания нового документа: Базовый путь

    В конструкторах нового документа можно задать Базовыйпуть.

    По действию он аналогичен заданию атрибута
    <…
    xmlns:xml="http://www.w3.org/XML/1998/namespace"
    xml:base="базовый URI" …

    Отличаясь тем, что пути задаются стандартным для Parser способом (см. «Приложение 1. Пути к файлам и каталогам»), что куда удобнее задания полного дискового пути, включающего путь к веб-пространству. По умолчанию равен пути к текущему обрабатываемому документу.
    Внимание: символ «/» на конце пути обязателен.


    Параметры преобразования документа в текст

    В ряде методов можно задать хешПараметры_преобразования_в_текст.

    Они идентичны атрибутам элемента .
    Исключением являются атрибуты doctype-public и doctype-system, которые так задать нельзя.
    Пока также является исключением cdata-section-elements.

    По умолчанию текст создается в кодировке $request:charset, однако в XML заголовке или в элементе meta для HTML-метода Parser указывает кодировку $response:charset. Такое поведение можно изменить, явно указав кодировку в или соответствующем параметре преобразования. [3.1.2]

    При создании объекта класса file можно задать параметр media-type, при задании нового тела ответа заголовок ответа content-type получит значение этого параметра.


    Parser://метод/параметр. Чтение

    Parser может считать XML из произвольного источника.
    Везде, где можно считать XML, можно задать адрес документа вида…
    parser://метод/параметр

    Считывание документа по такому адресу приводит чтению результата работы метода Parser, ^метод[/параметр].


    $document

    $document[^xdoc::create[document]]
    $paraNode[^document.createElement[para]]
    $addedNode[^document.documentElement.appendChild[$paraNode]]
    $response:body[^document.string[]]

    $response:body

    $document[^xdoc::create{

    текст
    }]
    $response:body[^document.string[]]

    ^xdoc::create

    $document[^xdoc::create{

    строка1

    строка2

    }]
    $response:body[^document.file[]]

    ^document.string

    # выдаст документ в HTML-представлении без отступов
    ^document.string[
    $.method[html]
    $.indent[no]
    ]

    ^document.save

    $document[^xdoc::create{

    строка1

    строка2

    }]
    ^document.save[saved.xml]

    $xdoc

    $xdoc[^xdoc::create{

    string

    }]
    $xdoc.search-namespaces.s[urn:special]
    ^xdoc.selectString[string(//s:code)]

    $.method

    $document[^xdoc::create{

    строка1

    строка2

    }]
    ^document.string[
    $.method[html]
    ]

    будет считан из каталога

    $sheet[^xdoc::create[/xsl/]{



    }]

    Здесь файл import.xsl, будет считан из каталога /xsl/.

    Подробная спецификация XPath доступна здесь:

    $d[^xdoc::create{


    }]
    # результат=список из двух элементов "t"
    $list[^d.select[/document/t]]
    # перебираем найденные листы:
    # этот код будет работать
    # даже если запрос не найдет ни одного листа
    ^for[i](0;$list-1){
    $node[$list.$i]
    Имя: $node.nodeName

    Тип: $node.nodeType

    }

    В Parser DOM-интерфейс NodeList - класс hash с ключами 0, 1, …

    Подробная спецификация XPath доступна здесь: http://www.w3.org/TR/xpath

    Подробная спецификация XPath доступна здесь:

    $d[^xdoc::create{
    }]
    ^if(^d.selectBool[/t/@n > 10]){
    /t/@n больше 10
    }{
    не больше
    }

    Подробная спецификация XPath доступна здесь: http://www.w3.org/TR/xpath

    Подробная спецификация XPath доступна здесь:

    $d[^xdoc::create{
    }]
    #результат=124
    ^d.selectNumber[number(/t/@n)+1]

    #результат=4
    ^d.selectNumber[2*2]


    Подробная спецификация XPath доступна здесь: http://www.w3.org/TR/xpath

    Подробная спецификация XPath доступна здесь:

    $d[^xdoc::create{
    }]
    # результат=один элемент "t"
    $element[^d.selectSingle[t]]
    # результат=2 (количество атрибутов )
    Количество атрибутов: ^element.attributes._count[]


    Подробная спецификация XPath доступна здесь: http://www.w3.org/TR/xpath

    Подробная спецификация XPath доступна здесь:

    $d[^xdoc::create{
    }]
    # результат=привет
    ^d.selectString[string(t/@attr)]

    Подробная спецификация XPath доступна здесь: http://www.w3.org/TR/xpath

    Пример хранения XSL шаблонов в базе данных

    @main[]

    # к этому моменту в $xdoc находится документ, который хотим преобразовать
    ^xdoc.transform[parser://xsl_database/main.xsl]

    @xsl_database[name]
    ^string:sql{select text from xsl where name='$name'}

    Причем относительные ссылки будут обработаны точно также, как если бы файлы читались с диска.
    Скажем, если parser://xsl_database/main.xsl ссылается на utils/common.xsl, будет загружен документ
    parser://xsl_database/utils/common.xsl, для чего будет вызван метод ^xsl_database[/utils/common.xsl].




    Пример (см. также «Урок 6. Работаем с XML»)

    # входной xdoc документ
    $sourceDoc[^xdoc::
    load[article.xml]]

    # преобразование xdoc документа шаблоном article.xsl
    $transformedDoc[^sourceDoc.transform[article.xsl]]

    # выдача результата в HTML виде
    ^transformedDoc.
    string[
    $.method[html]
    ]

    Если шаблон не считывается с диска, а создается динамически, важным вопросом становится «а откуда загрузятся ?», обратите внимание на возможность задания базового пути: «Параметр создания нового документа: Базовый путь».




    Пример загрузки XML-документа с диска

    $document[^xdoc::load[article.xml]]
    $response:body[^document.string[]]




    Пример загрузки XML-документа с HTTP-сервера

    $xdoc[^xdoc::load[http://www.cbr.ru/scripts/XML_daily.asp]]
    На
    ^xdoc.selectString[string(/ValCurs/@Date)]
    курс валюты
    $node[^xdoc.selectSingle[/ValCurs/Valute[CharCode='USD']]]
    "^node.selectString[string(Name)]"
    равен
    ^node.selectString[string(Value)]


    ^taint[^xdoc.string[]]





    Save. Сохранение документа в файл

    ^документ.save[путь]
    ^документ.save[путь;Параметры_преобразования_в_текст]

    Сохраняет документ в текстовый файл. Возможно задание параметров_преобразования в текст.
    По умолчанию создается XML-представление документа с заголовком (можно отключить вывод заголовка, задав соответствующий параметр).

    Путь - путь к файлу.


    Search-namespaces. Хеш пространств имен для поиска[3.1.2]

    $документ.search-namespaces

    Для использования префиксов пространств имен в методах xnode.select* необходимо заранее эти префиксы определить в данном хеше.

    Здесь
    ·ключи - префиксы пространств имен,
    ·значения - их URI.




    Select. XPath поиск узлов

    $NodeList[^узел.select[XPath-запрос]]

    Выдает список узлов, найденных в контексте узла по заданному XPath-запросу. Если запрос не вернул подходящих узлов, выдается пустой список.

    Для использования в запросе префиксов пространств имен необходимо их заранее определить, см. $xdoc.search-namespaces.


    SelectBool. Вычисление логического XPath запроса

    ^узел.selectBool[XPath-запрос]

    Выдает результат выполнения XPath-запроса в контексте узла, если это логическое значение. Если не логическое значение, выдается ошибка типа parser.runtime.

    Для использования в запросе префиксов пространств имен необходимо их заранее определить, см. $xdoc.search-namespaces.


    SelectNumber. Вычисление числового XPath запроса

    ^узел.selectNumber[XPath-запрос]

    Выдает результат выполнения XPath-запроса в контексте узла, если это число. Если не число, выдается ошибка типа parser.runtime.

    Для использования в запросе префиксов пространств имен необходимо их заранее определить, см. $xdoc.search-namespaces.


    SelectSingle. XPath поиск одного узла

    ^узел.selectSingle[XPath-запрос]

    Выдает узел, найденный в контексте узла по заданному XPath-запросу. Если запрос не нашел подходящего узла, выдается void. Если запрос выдал больше, чем один узел, выдается ошибка xml-типа.

    Для использования в запросе префиксов пространств имен необходимо их заранее определить, см. $xdoc.search-namespaces.


    SelectString. Вычисление строчного XPath запроса

    ^узел.selectString[XPath-запрос]

    Выдает результат выполнения XPath-запроса в контексте узла, если это строка. Если не строка, выдается ошибка типа parser.runtime.

    Для использования в запросе префиксов пространств имен необходимо их заранее определить, см. $xdoc.search-namespaces.


    String. Преобразование документа в строку

    ^документ.string[]
    ^документ.string[Параметры_преобразования_в_текст]

    Преобразует документ в текстовую форму. Возможно задание параметров_преобразования.
    По умолчанию создается XML-представление документа с заголовком (можно отключить вывод заголовка, задав соответствующий параметр).


    Transform. XSL преобразование

    ^документ.transform[шаблон]
    ^документ.transform[шаблон][XSLT-параметры]

    Осуществляет XSL-преобразование документа по шаблону. Возможно задание XSLT-параметров.

    Шаблон - или путь_к_файлу_с_шаблоном, или xdoc документ.
    Parser может считать XML из произвольного источника, см. раздел «Чтение XML из произвольного источника».

    XSLT-параметры
    - хеш строк, доступных из шаблона через .

    Внимание: Parser (в виде модуля к Apache или IIS) кеширует результат компиляции файла_с_шаблоном во внутреннюю форму, повторная компиляция не производится, а скомпилированный шаблон берется из кеша. Вариант CGI также кеширует шаблон, но только на один запрос. Шаблон перекомпилируется при изменении даты файлов шаблона.




    Выдача XHTML

    Если необходимо выдать XHTML, следует использовать такие атрибуты элемента :

    xmlns="http://www.w3.org/1999/xhtml"
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    >

    Обратите внимание на указание xmlns без префикса - так необходимо делать, чтобы все создаваемые в шаблоне элементы без префикса попадали в пространство имен xhtml. Необходимо задавать xmlns без префикса в каждом .xsl файле, этот параметр не распространяется на включаемые файлы.

    А также необходимо задать такие атрибуты :
    doctype-public="-//W3C//DTD XHTML 1.0 Strict//EN"
    doctype-system="DTD/xhtml1-strict.dtd"
    />

    Внимание: не задавайте атрибут method. XHTML это разновидность метода xml, включающаяся при использовании следующих doctype:
    -//W3C//DTD XHTML 1.0 Strict//EN
    -//W3C//DTD XHTML 1.0 Frameset//EN
    -//W3C//DTD XHTML 1.0 Transitional//EN




    А что это вообще такое?

    ClientCharset. Параметр подключения

    Параметр ClientCharset определяет кодировку, в которой необходимо общаться с SQL-сервером. Если параметр не указан, Parser считает, что общение с SQL-сервером идет в кодировке $request:charset.

    Список допустимых кодировок определяется в Конфигурационном файле.




    Для MySQL

    mysql://user:password@host:port/database?
    ClientCharset=кодировка& [3.1.2]
    timeout=3&
    compress=1&
    named_pipe=1&
    autocommit=1

    Необязательные параметры:
    Port - номер порта сервера баз данных. Можно использовать выражение:
    user:password@имя_хоста:номер_порта/database,

    А можно вместо имени_хоста и номера_порта передать путь к UNIX сокету в квадратных скобках (UNIX socket - это некий магический набор символов (путь), который вам расскажет администратор MySQL, если он - это не вы. Через этот сокет может идти общение с сервером):
    user:password@[/unix/socket]/database

    ClientCharset - задает кодировку, в которой необходимо общаться с SQL-сервером; [3.1.2]
    Timeout - задает значение параметра Connect timeout в секундах;
    Compress - режим сжатия трафика между сервером и клиентом;
    Named_pipe - использование именованых каналов для соединения с сервером MySQL, работающим под управлением Windows NT.



    Для ODBC

    odbc://DSN=dsn;UID=user;PWD=password;ClientCharset=кодировка

    ClientCharset - задает кодировку, в которой необходимо общаться с SQL-сервером. [3.1.2]

    Замечание: В коде Parser символ ";" в строке подключения к БД необходимо предварять символом "^".


    Для Oracle

    oracle://user:password@service?
    ClientCharset=кодировка& [3.1.2]
    LowerCaseColumnNames=0& [3.1.1]
    NLS_LANG=RUSSIAN_AMERICA.CL8MSWIN1251&
    NLS_DATE_FORMAT=YYYY-MM-DD HH24:MI:SS&
    NLS_LANGUAGE=language-dependent conventions&
    NLS_TERRITORY=territory-dependent conventions&
    NLS_DATE_LANGUAGE=language for day and month names&
    NLS_NUMERIC_CHARACTERS=decimal character and group separator&
    NLS_CURRENCY=local currency symbol&
    NLS_ISO_CURRENCY=ISO currency symbol&
    NLS_SORT=sort sequence&
    ORA_ENCRYPT_LOGIN=TRUE

    Если имена колонок в запросе select не взять в кавычки, Oracle преобразует их к ВЕРХНЕМУ регистру. По-умолчанию, Parser преобразует их к нижнему регистру. Указав параметр LowerCaseColumnNames=0 можно отключить преобразование в нижний регистр.

    ClientCharset - задает кодировку, в которой необходимо общаться с SQL-сервером. [3.1.2]

    Информацию по остальным параметрам можно найти в документации по «Environment variables» для Oracle. Однако, мы рекомендуем всегда задавать параметры NLS_LANG и NLS_DATE_FORMAT такими, какие указаны выше.


    Для PostgreSQL

    pgsql://user:password@host[:port]|[local]/database?
    ClientCharset=кодировка& [3.1.2]
    datestyle=ISO,SQL,Postgres,European,NonEuropean,US,German
    NonEuropean то же, что US
    по умолчанию ISO

    Необязательные параметры:
    port - номер порта.

    Можно задать:
    user:password@host:port/database,

    а можно:
    user:password@local/database

    В этом случае произойдет соединение с сервером, расположенным на локальной машине.

    ClientCharset - задает кодировку, в которой необходимо общаться с SQL-сервером; [3.1.2]

    Datestyle - если задан этот параметр, то при соединении с сервером драйвер выполнит команду:
    set DATESTYLE=значение.



    Переменная CLASS_PATH

    В конфигурационном методе может быть задана переменная или таблица CLASS_PATH, в которой задается путь (пути) к каталогу с файлами классов. Если имя подключаемого модуля - относительно, то файл ищется по CLASS_PATH, (если CLASS_PATH таблица, то каталоги в ней перебираются снизу вверх).

    Пример таблицы CLASS_PATH:

    $CLASS_PATH[^table::create{path
    /classes/common
    /classes/specific
    }]

    Теперь по относительному пути my/class.p поиск файла будет проходит в таком порядке:
    /classes/specific/my/class.p
    /classes/common/my/class.p




    Пути к файлам и

    Для доступа к файлам и каталогам в Parser можно использовать абсолютный или относительный путь.

    Абсолютный путь начинается слешем, а файл ищется от корня веб-пространства. Файл по относительному пути ищется от каталога, в котором находится запрошенный документ.

    Пример абсолютного пути:
    /news/archive/20020127/sport.html

    Пример относительного пути:
    относительно каталога /news/archive…
    20020127/sport.html

    При записи файлов необходимые каталоги создаются автоматически.

    Внимание: корень веб-пространства, переданный веб-сервером, можно изменить: см. «Корень веб-пространства».

    Внимание: Parser преобразует пути к языку file-spec (см. «Внешние и внутренние данные»).

    Также ряд методов…
    ·file::load
    ·table::load
    ·xdoc::load

    …может работать с внешними HTTP-серверами, если имя загружаемого документа содержит префикс
    http://

    Этим методам также можно задать дополнительные опции загрузки документа по HTTP, это хеш, ключами которого могут быть:

    Опция
    По-умолчанию
    Значение
    $.charset[кодировка]
    берется из заголовка HTTP-ответа
    Кодировка документов на удаленном сервере. В эту кодировку перекодируется строка запроса, и из этой кодировки перекодируется ответ. [3.1.0]
    $.timeout(секунд)
    2 секунды
    Время ожидания ответа HTTP сервера в секундах. Операция загрузки должна быть завершена за это время, иначе возникнет ошибка.
    $.method[HTTP-МЕТОД]
    GET
    $.form[
    $.поле[
    значение]
    $.поле[$таблица_значений]

    ]

    отсутствует
    Параметры запроса. Для GET запроса они будут переданы в ?строке_запроса. Для запросов с другим method, параметры будут переданы с
    Content-type: application/x-www-form-urlencoded
    Значением может являться строка или таблица строк из одного столбца.
    Нельзя передавать файлы.
    [3.1.2]

    Предпочтительно задавать параметры запросам именно при помощи $.forms, а не передавать их в ?параметрах самостоятельно.
    $.headers[
    $.HTTP-ЗАГОЛОВОК[
    значение]

    ]

    $.User-Agent[parser3]
    Хеш, содержащий дополнительные HTTP-заголовки, которые необходимо передать на HTTP-сервер

    Значение HTTP-заголовка может быть дата, строка или хеш с обязательным ключом value.
    Дата может использоваться и в качестве значения поля и в качестве значения атрибута поля, при этом она будет стандартно отформатирована.
    $.any-status(1)
    0
    Логическое: допустим ли статус ответа, не равный 200. Если ЛОЖЬ, и будет получен статус, не равный 200, возникнет системная ошибка http.status.
    [3.0.8]
    $.user[пользователь]
    отсутствует
    Задает параметры запроса к серверу,
    $.password[пароль]
    отсутствует
    использующему стандартную HTTP авторизацию.
    [3.1.2]
    Для ^file::load[…] также можно дополнительные опции загрузки [3.0.8], это хеш, ключами которого могут быть:

    Опция
    По-умолчанию
    Значение
    $.offset(смещение)
    0
    Загрузить данные начиная с этого смещения (в байтах).
    $.limit(ограничение)
    -1
    Загрузить не более данного количества байт.

    Форматные строки

    Форматная строка определяет форму представления значения числа. В общем случае она имеет следующий вид:

    %Длина.ТочностьТип

    Тип - определяет способ преобразования числа в строку.

    Существуют следующие типы:

    d
    - десятичное целое число со знаком

    u
    - десятичное целое число без знака

    o
    - восьмеричное целое число без знака

    x
    - шестнадцатеричное целое число без знака; для вывода цифр, больших 9, используются буквы a, b, c, d, e, f

    X
    - шестнадцатеричное целое число без знака; для вывода цифр, больших 9, используются буквы A, B, C, D, E, F

    f
    - действительное число
    Точность - точность представления дробной части, т. е. количество знаков после запятой. Если для отображения дробной части значения требуется больше знаков, то значение округляется. Обычно точность указывают в том случае, если используется тип преобразования f. Для других типов указывать точность не рекомендуется. Если точность не указана, то для типа преобразования f она по умолчанию принимается равной 6. Если указана точность 0, то число выводится без дробной части

    Длина - количество знаков, отводимое для значения. Может получиться так, что для отображения полученного значения требуется меньше символов, чем указано в блоке Длина. Например, указана длина 10, а получено значение 123. В этом случае слева к значению будет приписано семь пробелов. Если нужно, чтобы слева приписывались не пробелы, а нули, следует в начале блока Длина поместить 0, например, написать не 10, а 010. Блок Длина может отсутствовать, тогда для значения будет отведено ровно столько символов, сколько требуется для его отображения.

    Формат строки подключения оператора connect

    Строка подключения обрабатывается драйвером базы данных для Parser3.

    Perl-совместимые регулярные выражения

    Подробную информацию по Perl-совместимым регулярные выражениям (Perl Compatible Regular Expressions, PCRE) можно найти в документации к Perl (см. http://www.perldoc.com/perl5.6.1/pod/perlre.html), в документации к использованной в Parser библиотеке PCRE 2.09 (см. http://www.pcre.org/man.txt), а также в большом количестве специальной литературы, содержащей помимо всего остального много практических примеров. Особенно детально использование регулярных выражений описано в книге Дж. Фридла «Регулярные выражения» издательства «O'Reilly» (ISBN 1-56592-257-3), перевод книги на русский язык: издательство «Питер» (ISBN: 5-272-00331-4, второе издание; ISBN: 5-318-00056-8 первое издание).

    Краткое описание, которое приводится тут, имеет справочный характер.

    Регулярное выражение - это шаблон для поиска подстроки, который должен совпасть с подстрокой слева направо в строке поиска. Большинство символов в этом шаблоне представлены сами собою, и при поиске просто проверяется наличие этих символов в строке поиска в заданной последовательности. В качестве простейшего примера можно привести шаблон для поиска «шустрая лиса», который должен совпасть с аналогичным набором символов в строке поиска. Мощь регулярных выражений состоит в том, что помимо обычных символов, они позволяют включать в шаблоны альтернативные варианты выбора и повторяющиеся фрагменты с помощью метасимволов. Эти метасимволы ничего не значат сами по себе, но при использовании их в регулярных выражениях, они обрабатываются особым образом.

    Существует два различных набора метасимволов:

    1. Распознаваемые в любой части шаблона, не заключенной в квадратные скобки
    2. Распознаваемые в частях шаблона, заключенных в квадратные скобки

    К метасимволам, распознаваемым вне квадратных скобок относятся следующие:

    \

    общее обозначение для escape-последовательностей. Имеют различное использование, рассмотрены ниже

    ^

    совпадает с началом фрагмента для поиска или перед началом строки в многострочном режиме

    $

    совпадает с концом фрагмента для поиска или перед концом строки в многострочном режиме

    .

    символьный класс, содержащий все символы. Этот метасимвол, совпадает с любым символом кроме символа новой строки по умолчанию.

    [...]

    символьный класс. Совпадение происходит с любым элементом из заданного в квадратных скобках списка

    |

    метасимвол означающий «или». Позволяет объединить несколько регулярных выражений в одно, совпадающее с любым из выражений-компонентов

    (...)

    ограничение подстроки поиска в общем шаблоне поиска

    ?

    совпадает с одним необязательным символом

    *

    совпадает с неограниченным количеством любых необязательных символов

    +

    совпадает с неограниченным количеством символов. Для совпадения требуется хотя бы один произвольный символ

    {мин, макс}

    интервальный квантификатор - требуется минимум экземпляров, допускается максимум экземпляров.


    Часть шаблона, заключенная в квадратные скобки называется символьным классом. В описании символьного класса можно использовать только следующие метасимволы:

    \

    Escape - символ

    ^

    Инвертированный символьный класс, мета-символ обязательно должен быть первым символом в описании класса. Совпадение будет происходить с любыми символами, не входящими в символьный класс

    -

    Используется для обозначения интервала символов

    [...]

    Ограничитель символьного класса


    Использование метасимвола «\».

    Обратный слеш имеет несколько вариантов использования. В случае если вслед за ним следует символ, не обозначающий букву алфавита, обратный слеш выполняет функцию экранирования и отменяет специальное значение, которое может иметь этот символ. Такое использование этого метасимвола возможно как внутри символьного класса, так и вне его. В качестве примера, если необходимо найти символ «*», то используется следующая запись в шаблоне «\*». В случае необходимости экранировать сам символ «\» используется запись «\\».

    Второй вариант использования этого мета-символа - для описания управляющих символов в шаблоне. Можно использовать следующие escape-последовательности:

    сигнал
    \cx «control-x», где х - любой символ
    ASCII-символ escape
    \f подача бумаги
    \n новая строка
    \r возврат курсора
    \t табуляция
    \xhh шестнадцатиричный код символа hh
    \ddd восьмиричный код символа ddd

    Третий вариант - для определения специфических символьных классов

    \d любое десятичное число [0-9]
    \s пропуск, обычно [ \f\n\r\t] Первый символ квадратных скобках - пробел
    \w символ слова, обычно [a-zA-Z0-9_]
    \D \S \W отрицание \d \s \w

    Четвертый вариант - для обозначения мнимых символов. В PCRE существуют символы, которые соответствуют не какой-либо литере или литерам, а означают выполнение определенного условия, поэтому в английском языке они называются утверждениями (assertion). Их можно рассматривать как мнимые символы нулевого размера, расположенные на границе между реальными символами в точке, соответствующей определенному условию. Эти утверждения не могут использоваться в символьных классах (\b имеет дополнительное значение и обозначает возврат каретки внутри символьного класса)

    \b граница слова
    \B отсутствие границы слова
    \A «истинное» начало строки
    \Z «истинный» конец строки или позиция перед символом начала новой строки,
    расположенного в «истинном» конце строки
    \z «истинный» конец строки

    Как правильно назначить

    Имя должно быть понятно как минимум вам самим, а как идеал - любому человеку, читающему ваш код. Имя может быть набрано русскими или латинскими буквами, главное - единообразие. Рекомендуем все же пользоваться английским (а вдруг вас ждет мировое признание?). Слова в именах лучше использовать в единственном числе. Если есть необходимость - пользуйтесь составными именами вида column_color. Глядя на такое имя можно сразу понять, что оно означает.

    Parser чувствителен к регистру!

    $Parser и $parser - разные переменные!

    Есть определенные ограничения на использование в именах символов. Для Parser имя всегда заканчивается перед:
    пробелом
    табуляцией
    переводом строки
    ; ] } ) " < > # + * / % & | = ! ' , ?
    в выражениях заканчивается и перед "-"

    Код:
    $var[значение_из_переменной]
    $var>text


    выдаст на экран:
    значение_из_переменной>text

    т.е. символ '>' Parser считает окончанием имени переменной $var и подставляет ее значение, поэтому вышеуказанные символы не следует использовать при составлении имен.

    Если есть необходимость сразу после значения переменной (т.е. без пробела, который является концом имени) вывести символ, который не указан выше (например, нам нужно поставить точку сразу после значения переменной) используется следующий синтаксис:
    ${var}.text

    даст:
    значение_из_переменной.text

    Нельзя (!) пользоваться в именах символами ".", ":", "^" поскольку они будут расцениваться как часть кода Parser, что приведет к ошибкам при обработке вашего кода.

    Все остальные символы использовать в именах, в принципе, можно, но лучше всего отказаться от использования в именах каких-либо служебных и специальных символов кроме случаев крайней необходимости (в практике не встречаются), за исключением знака подчеркивания, который не используется Parser и достаточно нагляден при использовании в именах.

    Как бороться с ошибками

    Для начала вдумчиво прочитайте сообщение об ошибке. В нем содержится имя файла, вызвавшего ошибку и номер строки в нем. Здесь требуется внимательность при написании кода и справочник. Всегда помните о том, что Parser оперирует объектной моделью, поэтому внимательно следите за тем, с объектом какого класса вы работаете. Некоторые методы возвращают объекты других классов!

    Так, например, некоторые методы класса date возвращают объект класса table. Попытка вызвать для этого объекта методы класса date приведет к ошибке. Нельзя вызывать методы классов для объектов, которые к этим классам не принадлежат. Впрочем, этот этап вам удастся преодолеть довольно быстро. Еще одна категория ошибок - ошибки в логике работы самого кода. Это уже сложнее, и придется запастись терпением. Обязательно давайте грамотные имена переменным, методам, классам и комментируйте код.

    Если и в этом случае не удастся понять причины неверной работы - попробуйте обратиться к справочнику. «Если ничего не помогает - прочтите, наконец, инструкцию…» Последняя стадия в поиске ошибок - вы близки к сумасшествию, пляшете вокруг компьютера с бубном, а код все равно не работает. Здесь остается только обратиться за помощью к тем, кто пока разбирается в Parser чуть лучше, чем вы. Задайте свой вопрос на форуме, посвященном работе на этом языке и вам постараются ответить. Вы не одиноки! Удачи вам!

    в вашей базе хранятся

    Допустим данные в вашей базе хранятся в кодировке windows-1251, строку подключения стоит написать так:
    odbc://DSN=dsn^;UID=user^;PWD=password^;ClientCharset=windows-1251

    в вашей базе хранятся

    Допустим данные в вашей базе хранятся в кодировке windows-1251, строку подключения стоит написать так:
    oracle://user:password@service?ClientCharset=windows-1251&NLS_LANG=RUSSIAN_AMERICA.CL8MSWIN1251&NLS_DATE_FORMAT=YYYY-MM-DD HH24:MI:SS

    

        Бизнес в интернете: Сайты - Софт - Языки - Дизайн