• Главная
  • Контакты
Подписаться:
  • Twitter
  • Facebook
  • RSS
  • VK
  • PushAll
logo

logo

  • Все
    • Положительные
    • Отрицательные
  • За сегодня
    • Положительные
    • Отрицательные
  • За вчера
    • Положительные
    • Отрицательные
  • За 3 дня
    • Положительные
    • Отрицательные
  • За неделю
    • Положительные
    • Отрицательные
  • За месяц
    • Положительные
    • Отрицательные
  • За год
    • Положительные
    • Отрицательные
  • Сортировка
    • По дате (возр)
    • По дате (убыв)
    • По рейтингу (возр)
    • По рейтингу (убыв)
    • По комментам (возр)
    • По комментам (убыв)
    • По просмотрам (возр)
    • По просмотрам (убыв)
Главная
  • Все
    • Положительные
    • Отрицательные
  • За сегодня
    • Положительные
    • Отрицательные
  • За вчера
    • Положительные
    • Отрицательные
  • За 3 дня
    • Положительные
    • Отрицательные
  • За неделю
    • Положительные
    • Отрицательные
  • За месяц
    • Положительные
    • Отрицательные
  • Главная
  • «Мечта лентяя» или скриптовый движок на самом себе

«Мечта лентяя» или скриптовый движок на самом себе +11

24.04.2017 07:16
EasyLy 9 3800 Источник
C#*, .NET*
У разработчиков прикладного ПО очень часто возникает потребность встроить в свой продукт некий скриптовый язык, который бы решал часть задач, не описанных детально на момент проектирования системы. Действительно удобно: и возможность расширения функциональности есть, и трудоёмкость создания такого решения, на первый взгляд, невелика.

Эту давнюю мечту можно было бы назвать «мечтой лентяя», если бы имеющиеся общедоступные встраиваемые скриптовые средства были бы просты. Готовые средства существовали давно, например на платформе Windows, ещё в прошлом веке можно было использовать интерфейсы VBScript и Jscript через COM-интерфейс IActiveScriptSite. В настоящее время существует большое количество и других решений, например на базе Lua, но все они имеют одну неприятную особенность, сильно ограничивающую желание их применять.

Скрипты прекрасно работают и сами по себе, на них можно выполнять и логику, и арифметику, но пользы от них ровным счётом никакой, если сложно или нет возможности:

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

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

Прикладное ПО сейчас очень часто пишется на C#, и хотелось бы иметь что-то знакомое, но гибкое, и позволяющее писать скрипты. Такое решение есть, и оно заслуживает пристального внимания. Это пространство имён System.CodeDom.Compiler с его классом CSharpCodeProvider. Всё это появилось ещё в .NET 4.0, но по какой-то причине в большинстве публикаций по C# не затрагивался вопрос написания скриптов на C#, используя сам же язык C# в качестве базового. А это очень и очень удобно для написания и дальнейшего сопровождения продукта.

В этом случае самый главный и интересный метод — CompileAssemblyFromSource(), который выполняет компиляцию, генерирует сообщения об ошибках, и мы уже запросто можем написать «Hello world!»

using System;
using System.IO;
using Microsoft.CSharp;
using System.CodeDom.Compiler;
using System.Reflection;
using System.Text;

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            // готовим текст скрипта
            StringBuilder sb = new StringBuilder();
            sb.AppendLine("using System;");
            sb.AppendLine("namespace ConsoleApplication1");
            sb.AppendLine("{");
            sb.AppendLine("   public class MyScripter");
            sb.AppendLine("   {");
            sb.AppendLine("       public void Hello()");
            sb.AppendLine("       {");
            sb.AppendLine("             Console.WriteLine(\"Hello world!\");");
            sb.AppendLine("       }");
            sb.AppendLine("   }");
            sb.AppendLine("}");

            // компилируем
            CSharpCodeProvider codeProvider = new CSharpCodeProvider();
            CompilerResults compileResults = codeProvider.CompileAssemblyFromSource(
                       new CompilerParameters(), new string[] { sb.ToString() });

            // выводим ошибки, если они есть
            foreach (CompilerError err in compileResults.Errors)
                Console.WriteLine("Error({0:1}): {2} {3}", err.Line, err.Column, 
                                  err.ErrorNumber, err.ErrorText);
            if (compileResults.Errors.HasErrors) return;

            // загружаем получившуюся dll в память
            byte[] dllBytes = File.ReadAllBytes(compileResults.PathToAssembly);
            Assembly asmDll = Assembly.Load(dllBytes, null);
            Type objType = asmDll.GetType("ConsoleApplication1.MyScripter");

            // создаём объект класса из скрипта
            object oClassInst = Activator.CreateInstance(objType);

            // получаем точка входа и выполняем её
            MethodInfo entry = objType.GetMethod("Hello", new Type[] {});
            entry.Invoke(oClassInst, null);
        }
    }
}

Запускаем на исполнение:

image

Итак, в простейшем виде скрипт на C# успешно работает. Простым изменением текста скрипта мы можем влиять на его работу. Собственно, осталось лишь передать в скрипт в качестве примера какой-либо объект из основной программы.

В качестве такого объекта вполне подойдёт объект типа string:

using System;
using System.IO;
using Microsoft.CSharp;
using System.CodeDom.Compiler;
using System.Reflection;
using System.Text;

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            string sMyStr = "Before Script.";

            // готовим текст скрипта
            StringBuilder sb = new StringBuilder();
            sb.AppendLine("using System;");
            sb.AppendLine("namespace ConsoleApplication1");
            sb.AppendLine("{");
            sb.AppendLine("   public class MyScripter");
            sb.AppendLine("   {");
            sb.AppendLine("       public void Hello(ref string s)");
            sb.AppendLine("       {");
            sb.AppendLine("             Console.WriteLine(\"Hello world!\");");
            sb.AppendLine("             s=\"After Script.\";");
            sb.AppendLine("       }");
            sb.AppendLine("   }");
            sb.AppendLine("}");

            // компилируем
            CSharpCodeProvider codeProvider = new CSharpCodeProvider();
            CompilerResults compileResults = codeProvider.CompileAssemblyFromSource(
                     new CompilerParameters(), new string[] { sb.ToString() });

            // выводим ошибки, если они есть
            foreach (CompilerError err in compileResults.Errors)
                Console.WriteLine("Error({0:1}): {2} {3}", err.Line, err.Column, 
                                   err.ErrorNumber, err.ErrorText);
            if (compileResults.Errors.HasErrors) return;

            // загружаем получившуюся dll в память
            byte[] dllBytes = File.ReadAllBytes(compileResults.PathToAssembly);
            Assembly asmDll = Assembly.Load(dllBytes, null);
            Type objType = asmDll.GetType("ConsoleApplication1.MyScripter");

            // создаём объект класса из скрипта
            object oClassInst = Activator.CreateInstance(objType);

            // получаем точка входа и готовим параметры
            MethodInfo entry = objType.GetMethod("Hello", 
                          new Type[] { typeof(string).MakeByRefType() });
            Object[] param = new Object[] { sMyStr };

            Console.WriteLine(param[0]);        // до выполнения скрипта
            
            entry.Invoke(oClassInst, param);    // вызов метода

            Console.WriteLine(param[0]);        // после выполнения скрипта
        }
    }
}

Запускаем на исполнение:

image

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

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

Недостатком подхода можно считать только то, что при компиляции создаётcя dll во временном каталоге ОС.

Путь разрешения этого недостатка ведёт нас в сторону использования пространства System.Reflection.Emit, но это достаточно объёмный материал, подходящий для отдельной статьи. Это сложно, т. к. в данном случае компилятор и генерацию придется писать самостоятельно. Но зато какие возможности по придумыванию своего собственного синтаксиса! Да и назвать новый язык программирования можно в честь себя или любимой кошки.
Удачи!

Аркадий Пчелинцев, архитектор проектов
Поделиться с друзьями
-->

Комментарии (9)


  1. ColdPhoenix
    24.04.2017 10:21
    #10188124
    +4

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


    1. EasyLy
      24.04.2017 16:28
      #10188852

      Roslyn – хорошая штука, но требует .Net 4.5.2. Ниже он не соберётся. Да и великоват он
      для простых задач.


  1. molnij
    24.04.2017 10:44
    #10188158
    +8

    Ох сколько вам открытий чудных…
    Во-первых, если вы будете исполнять скрипты чуть больше чем hello world, вдруг выяснится отличие интерпретируемых от компилируемых языков — фаза компиляции ни разу не бесплатная. И если у интерпретируемых она естественным образом встроена в воркфлоу, то у шарпа вполне можно огрести заметные тормоза на ней.
    Потом обнаружится, что «гадить» в общий домен временными сборками скрипта — идея так себе, особенно если вспомнить, что удалить однажды загруженную сборку из домена невозможно (это слегка коррелирует с наличием временных файлов .dll, но проблема совсем в другом). А если делать по-науке и плодить домены, то придется изрядно позаниматься любовью с передачей нетривиальных параметров между доменами.
    А статей на эту тему достаточно. Даже на хабре поиск по CSharpCodeProvider выдает вполне адекватную подборку.


    1. Sinatr
      25.04.2017 10:46
      #10189830

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

      Полезность примера выводящего «Hello world» в данном случае тоже близка с 0.

      Единственно что обьяснили — это как скормить исходник и получить список ошибок, а это, извините, на статью не тянет, тема не раскрыта вообще от слова «полностью».

      Я в свое время (эмм… десять лет назад) использовал CodeDOM для пре-компиляции формул (вызывался метод, тело которого пользователь писал на C#, методу передавался список входных/выходных параметров, которые можно было использовать в вычислениях и логике и возвращать результат), с появление Roslyn это конечно смешно (хотя уверен новичкам сгодилось), и тут эта статья…


  1. ApeCoder
    24.04.2017 10:46
    #10188164

    https://blogs.msdn.microsoft.com/cdndevs/2015/12/01/adding-c-scripting-to-your-development-arsenal-part-1/


  1. kuda78
    24.04.2017 11:20
    #10188234
    +1

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

    Необходимо реализовать у объекта интерфейс IDispatch. В C# с использованием рефлексии реализация доступа к методам и свойствам будет элементарен.

    проводить синтаксический контроль исходного скрипта и генерировать сообщения о синтаксических ошибках

    А чем этот момент не устроил в ActiveScriptSite? В ошибке будет указан и номер строки и что за ошибка.

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

    С этим вопросом тоже все очень просто.
    https://docs.microsoft.com/en-us/scripting/winscript/active-script-debugging-overview


  1. ifle
    24.04.2017 12:25
    #10188330

    Вроде бы ClearScript самое оно
    https://clearscript.codeplex.com/


  1. mxms
    24.04.2017 12:45
    #10188372

    В этом отношении идеальный вариант это web-сервер на Forth, конфигурируемый Forth и со скриптингом на Forth.


  1. hensew
    24.04.2017 14:28
    #10188584

    Тоже бодался с этой темой, на Хабре в подходящее время вышла статья по NiL.JS
    Javascript удобнее для скриптов н никаких проблем с выгрузкой старого кода.

МЕТКИ

  • Хабы
  • Теги

.NET

C#

c#

СЕРВИСЫ
  • logo

    CloudLogs.ru - Облачное логирование

    • Храните логи вашего сервиса или приложения в облаке. Удобно просматривайте и анализируйте их.
Все публикации автора
  • Понимающий робот: голосовой интерфейс на ESP32 и OpenAI +6

    • 19.05.2025 06:30

    Ода структурам на примере работы с блоком питания Fnirsi DPS 150 +6

    • 24.01.2025 08:35

    Рассматриваем циклы процессора в контроллере CH32x035 +34

    • 14.01.2025 08:06

    Детали протокола управления блоком питания Fnirsi DPS-150 +18

    • 10.01.2025 08:54

    Применение нейросетей в DIY +20

    • 25.12.2024 11:54

    Стабилизируем положение потенциометра, считанное с АЦП +39

    • 14.11.2024 09:01

    Анализируем протокол управления блоком питания Fnirsi DPS 150 +28

    • 29.10.2024 08:39

    Будущее наступило, восстание машин уже идёт. Просто ползучим методом +25

    • 11.10.2024 08:46

    В погоне за скоростью. Оптимизация нейросетевых вычислений на процессоре К1967ВН044 компании «Миландр»

    • 26.06.2024 09:03

    Почему может не работать отладка на плате XIAO BLE

    • 19.06.2024 08:28

Подписка


ЛУЧШЕЕ

  • Сегодня
  • Вчера
  • Позавчера
01:49

В проблемах обнаружения принтеров вините драйверы Wi-Fi (и mDNS) +32

07:05

Окситоцин — гормон любви. А дружбы? +10

06:47

Смертельные ловушки для грабителей могил в древних гробницах. Где грань между правдой и вымыслом? +10

07:01

Продвинутый тестер автомобильных аккумуляторов и бортовой сети Konnwei KW650: часть 2 +7

03:48

На Луну и дальше. Компания SpaceX обновила концепцию лунного посадочного модуля +6

07:15

Requestum: CQRS-библиотека для .NET как альтернатива MediatR +5

06:00

Как упорядочить инфраструктуру в Cloud Director. Практическое руководство по metadata и тегам. Часть 2 +4

05:15

ИТ-аутсорсинг или ИТ-отдел? Как выбрать подрядчика и не переплатить +4

00:36

Многоразовые тестовые краевые соединители для печатных плат +4

07:00

Накрутка опыта, дурдом в IT найме и жизнь в постоянном обмане +3

05:43

Обзор проблем и решений в ризонинговых LLM. Часть 2 +3

08:36

Мы опубликовали silero-tts v5 на русском языке +2

08:07

Опрос для тех, кто в тяжёлой цифре, и тех, кто про неё думает +2

08:01

В темной-темной комнате пропадали важные-важные файлы… +2

06:53

Как писать промты для генерации кода: примеры для Python, JS и SQL +2

05:22

ERC-6909 vs ERC-1155: простой, как три копейки +2

08:31

Знакомимся, ссоримся, побеждаем: как работает групповая динамика в команде проекта +1

08:05

Как мы обучали дизайнеров VK проводить немодерируемые тесты. Сказ о разработке программы +1

07:39

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

07:07

AI‑поиск по патентам от Perplexity: для чего использовать? +1

07:01

Почему тормозят AMD Epyc +67

05:35

Как я оптимизировал отдел и получил врагов вместо премии +38

10:01

Спокойствие админа: автоматический мониторинг и Telegram-уведомления с Monit +35

13:01

По мотивам Хабра: рейтинг контента и пользователей на основе офелократии. Часть 1 +34

08:00

«Проклятый DDoS!»: защищаем сеть с помощью групп безопасности +34

08:47

Культовые программы нулевых: куда исчезли Winamp, ICQ, Nero и другие наши любимцы +32

20:12

Пробовали писать код с закрытыми глазами? Я делаю это всю жизнь: 32 вопроса незрячему программисту +31

11:54

Подстраиваемся под новые требования Google: переходим на 16 KB memory page (гайд для разработчиков React Native) +28

13:00

Как поднять n8n-сервер для автоматизации бизнес-процессов? Deep dive into Terraform +27

12:16

Я проверил 7 мифов о прочности домов — и понял, где нас обманывают +24

13:05

Вся история МКС в реальном времени: 25 лет на орбите день за днем +23

08:14

Автодополнение кода на примере YQL в YDB CLI +21

11:55

Как создавали уникальный советский синтезатор АНС и что с ним сейчас +20

11:18

Кто быстрее: исследую производительность std::format +20

11:08

Космос как стартап: NASA приземлила свой код на GitHub +20

08:14

Автодополнение кода на примере YQL в YDB CLI +20

07:00

Маленькая RAM vs новые фичи: как мы повышаем детализацию Карт и сохраняем стабильность +19

05:25

Перехватываем ИК-управление, чтобы сделать тупое устройство умным +18

09:43

Спрос на инженеров в России достиг рекорда — но молодые уже не соглашаются на 70 тысяч +17

05:16

Обучение и развитие: почему Linux — лучший выбор для детского ПК +16

06:23

Живешь в Европе? Будь готов к уголовному делу за счет в российском санкционном банке +84

07:05

Как музыканты издеваются над программистами +82

08:17

Я решал LeetCode 600 дней подряд и что из этого вышло +75

09:01

Музыкальные программы на игровых приставках +55

16:45

Pimp my RAM или превращаем оперативную память в фотографическую +54

13:41

Как одна бумажка 1942 года до сих пор определяет, на чём мы летаем +45

13:01

Кен Томпсон о взломе замков и рождении Unix +43

08:06

Сура ПК8000. Компьютер и жизнь сообщества разработчиков спустя почти 40 лет +41

15:16

Введение в Xray +38

07:26

Один сервер, вся сеть: руководство по RADIUS для новичков +35

05:33

Как я нашел криптографический бэкдор в крупнейшем даркнет-рынке: разбираем кражу $12 миллионов через уязвимость ECDSA +29

12:00

Бюджетная астрофотография +26

08:00

Atlas для Яндекса: как аналитики смотрят на карту, проверяют гипотезы и не тонут в данных +26

11:18

Как работает Docker под капотом +19

18:13

CPython простыми словами: всё, что нужно знать начинающему +18

13:03

Как я за вечер превратил хаос статистики Хабра в дашборд и создал по нему контент-план +18

12:43

Топ-10 нейросетей для написания песен и генерации музыки +16

10:52

Осенние будни DevOps: Debian 13 и Proxmox VE 9.0 в продакшене HOSTKEY +16

05:23

IT-сектор после перегрева, часть 2: наплыв нейросетей, бесконечные собеседования и конфликт интересов +16

12:15

Я изучил Discord подростков — и понял, почему им там лучше, чем на улице +15

ОБСУЖДАЕМОЕ

  • Живешь в Европе? Будь готов к уголовному делу за счет в российском санкционном банке +84

    • 442   38000

    Маркетплейсы всё? Продавцы переходят на прямые продажи +15

    • 284   64000

    Спрос на инженеров в России достиг рекорда — но молодые уже не соглашаются на 70 тысяч +17

    • 197   50000

    Я решал LeetCode 600 дней подряд и что из этого вышло +75

    • 130   40000

    Культовые программы нулевых: куда исчезли Winamp, ICQ, Nero и другие наши любимцы +32

    • 97   9600

    Как одна бумажка 1942 года до сих пор определяет, на чём мы летаем +45

    • 65   13000

    Как музыканты издеваются над программистами +82

    • 61   11000

    Я проверил 7 мифов о прочности домов — и понял, где нас обманывают +24

    • 56   11000

    Независимая радиосеть на базе смартфонов для децентрализованного интернета -2

    • 51   15000

    Тотальное протезирование в стоматологии +13

    • 46   1500

    Почему тормозят AMD Epyc +67

    • 45   9600

    Вакуумный дирижабль: Святой Грааль аэронавтики на грани возможного -2

    • 44   3400

    Осенние будни DevOps: Debian 13 и Proxmox VE 9.0 в продакшене HOSTKEY +16

    • 31   3300

    IT-сектор после перегрева, часть 2: наплыв нейросетей, бесконечные собеседования и конфликт интересов +16

    • 31   14000

    Техническое интервью — диалог или игра во власть? +15

    • 29   7500
  • Главная
  • Контакты
© 2025. Все публикации принадлежат авторам.