Skip to content

Latest commit

 

History

History
101 lines (82 loc) · 8.24 KB

README.md

File metadata and controls

101 lines (82 loc) · 8.24 KB

ВНИМАНИЕ! Реально работает только javascript версия и немного esp8266. Остальное в зачаточном состоянии.

Данный проект - попытка реализовать язык программирования Trac Кельвина Муэрса.

Кельвин Муэрс более известен в среде хранения и доступа к данным. Он сформулировал Закон Муэрса: «Степень использования информации прямо пропорционально легкости ее получения». «Система получения информации окажется невостребованной, если обладание информацией будет вызывать у клиента больше неудобств и беспокойства, чем ее отсутствие.»

Концепция языка родилась в 1959, первая реализация была осуществлена в 1964 году на PDP-11 в сотрудничестве с Питером Дойчем. В 1965 вышла статья Муэрса с описанием минимального языка и его интерпретатора (TRAC, A Procedure-Describing Language for the Reactive Typewriter) В интернете эта статья есть, однако в ней есть существенные ошибки и относиться к ней надо с аккуратностью - включив голову: http://web.archive.org/web/20050215092544/http://tracfoundation.org/trac64/procedure.htm мой вольный перевод этой статьи лежит в папке doc. Сайт сообщества Trac сейчас не доступен (ранее был), однако он сохранился в архивах:
http://web.archive.org/web/20050213111043/http://tracfoundation.org В дальнейшем язык активно развивался пока не столкнулся со стандартной для Америкацев проблемой права. Муэрсу пришлось судиться с сообществом R.E.S.I.S.T.O.R.S. которое продвигало похожий по концепции язык SAM-76. В результате оба языка почивали в бозе. Лично я познакомился с Trac в книге "Этюды для программистов" Чарльза Уэззерела.

Но это все лирика. Нафига я этим занялся? Дело в том, что Trac славится тем что у него САМЫЙ ПРОСТОЙ интерпретатор. После BraynFuck конечно. Но BF это уже извращение, а Trac - вполне себе макроязык. Описание интерпретатора есть в той самой статье в интернете. Кое что придется подчерпнуть и из соседних описаний (например формат записи в базе forms). То есть реализовано из чистого любопытства, безо всякой нужды. Вообще, прикольно видеть Json в языке 1964 года :) еще прикольнее видеть в нем Unicode c русскими именами команд и строк. Реализация вроде как "обратно совместимая", то есть все исходные программы по идее на ней должны работать. Под занавес: не пугайтесь синтаксиса. Примеры с моей точки зрения выбраны крайне не удачно, да еще и ошибки в них есть. На самом деле это обычный макроязык: идет-идет куча обычного текста, в нужном месте - вставка макроязыка - потом опять куча текста и так далее... Годится например для массированной впечатки адресов и обращений в письма (не даром: Reactive Typewriter - 1965 год), автоматизированного перевода. А если серьезно, то на нем можно даже web-сервер соорудить. Ну производительность то понятно... интерпретатор-же.

Краткие пояснения по языку:

Язык работает со строками. Других типов данных нет. Арифметика тоже работает именно со строковыми представлениями чисел. Язык это набор функций. Вид функций: #(имяфункции,0-или-больше-аргументов-через-запятую). Пробелы имеют значение. То есть #(ab) и #( ab) это две разные функции. То же и с аргументами. Текст вне функций напрямую передается на выходное устройство...но не без фокусов. То есть программа "Hello world#(ps,!)" напечатает именно то что мы и ожидаем, но не так как мы ожидаем: "!Hello world".
Напрягает? Нет проблем, напишите так: "Hello world `#(ps,!)`" или, что правильнее: "#(ps,Hello world)#(ps,!)" и все будет на местах. Мета-символ позволяет наводить порядок в записях. К слову. Кодировка изначально ASCII, можно использовать только первые 127 символов. Остальные просто игнорируются (за исключением неких спецсимволов). Также во входном потоке игнорируются CR, LF и TAB. Но! Поскольку изначально реализовано оно у меня на F#, а там сплошной Юникод, я не стал отказывать себе в удобствах и работаю в Юникоде. А потому - русский (да хоть хинди!) текст, русские имена функций и переменных - к вашим услугам. В качестве спецсимволов использую гендерные символы марса и венеры, какую-то арабскую скобку, и две вертикальные черты из псевдографики. Думаю вы не часто их используете в реальном тексте. Но есть символы с которыми в тексте приходится быть аккуратнее: "#(" - ясно это признак перехода на макроязык, "(...)" - тоже ясно, "," - внутри строк макроязыка будет восприниматься как переход к следующему параметру, "{" и "}" - нельзя использовать в запоминаемых в хранилище на диске данных и функциях - там JSON, он этого не простит. Я обычно просто перекодирую их во входных текстах и текстах программ куда нибудь в '\ufffx', а потом перекодирую полученную строку обратно. Если записать #(ab)#(cd) две функции вызовутся последовательно слева направо, а их результаты в том же порядке отпечатаны в выходном потоке. Если записать #(ab,#(cd)) то сначала вызовется cd, вернет строку, а эта строка передастся как аргумент к ab которая выполнится и ее выходная строка будет записана выходной поток. Если записать #(ab,(#(cd))) то немного сложнее. Экранирование. Те есть #(ps,#(ad,1,1)) это не тоже что #(ps,(#(ad,1,1))). Средние скобки как бы скрывают внутреннее выражение от выполнения поэтому первое выражение выдаст на печать "2", а второе выражение "#(ad,1,1)". Эта фишка применяется при определении именованных функций и обработке исключений. Так #(ps,##(np,(#(dv,4,2,#(ps,Ошибка деления)0)))) ##(np,) в нейтральном режиме, вернет "#(dv,4,2,#(ps,Ошибка деления)0)", а вот при #(np,)в активном режиме #(ps,#(np,(#(dv,4,2,#(ps,Ошибка деления)0)))) результат будет "2". Веселее другое, в нейтральном режиме: ##(ps, result=##(dv,10,1,(##(ps,error:)9999))) вернет... " result=10", все ясно, НО если при выполнении произошла ошибка, например деление на ноль: ##(ps, result=##(dv,10,0,(##(ps,error:)9999))) функция автоматически становится активной и выполняет код (##(ps,error:)9999) в результате "error: result=9999". То есть все функции принимающие параметр Z и вернувшие его из-за ошибки автоматически становятся активными. Я специально внедрил в код строки данных чтобы показать как это делается. Вообще, разница между активными (#) и нейтральными (##) функциями проявляется только при работе с кодом: активные функции вычислив значение отправляют его опять на выполнение а нейтральные - нет. По отношению к данным ( которые не являются сами кодом) они ведут себя одинаково не зависимо от количества '#'.

Свои макросы объявлять тоже можно просто сохраняя их в forms через #(ds,имя,(тело)) как данные, после этого их надо немного "подшлифовать напильником" с помощью #(ss,имя1парам,имя2парам...) а вызываются они через #(cl,имя,подставляемые,параметры) - cl должна быть в активном режиме! Будет работать и #(имя,подставляемые,параметры), но внутри определения рекурсивных функций надо использовать полную запись ;) Давайте подменим функцию печати #(ps,X):
#(ds,печать,(#(ps,X)))#(ss,печать,X))) после выполнения этого кода #(печать,русский-текст) напечатает "русский-текст". Макросы и переменные можно сохранять на диске а потом загружать обратно в хранилище форм.

Типовые операции:

присваивание константы var name = value : #(ds,name,value)`

использование переменной name : #(name)

инкремент переменной name++ : #(ds,name,#(ad,#(name),1)`

присваивание результата выражения name <- expression : #(ds,name,expression)`

определение функции func name(param1, param2) <- expression : #(ds,name,(expression))#(ss,param1,param2)`

вызов функции name(param1,param2) : #(name,param1,param2)`

условное ветвление if(condition,trueexpression,falseexpression) : #(eq,condition,0,falseexpression,trueexpression)`

цикл по условию while(condition, expression) : #(ds,tempname,(#(eq,condition,0,expression#(tempname),)))`#(tempname)`#(dd,tempname)`

цикл по счетчику for(startcount,endcount, expression) : #(ds,tempvar,(startcount))#(ds,tempname,(#(eq,tempvar,endcount,,expression#(ds,tempvar,#(ad,#(tempvar),1)))#(tempname))))`#(tempname)`#(dd,tempvar,tempname)`

Замечания по текущей реализации на JS:

  • не реализованы диагностические функции - не придумал еще как удобнее сделать трассировку
  • не работает и не будет работать сохранение на диск, позднее вместо него организуем хранение в localstorage
  • не работает работа с битами. Можно конечно потом сделать ее, однако реализаци битовых операций в языке не самая удачная и может быть будет правильнее реализовать их как то по другому.
  • добавлен преобразователь из языка простейшего функционального калькулятора в trac через RPN