Время создания
Фильтры

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



OnAppEnd из AppEventListenerBase на Windows 7.17.4 работает, а на Net.Core 7.17.4 НЕ РАБОТАЕТ (или там нету UserConnection, что в принципе не позволит вам сделать какую-либо операцию на уровне приложения). Вы никаким образом не сможете дать выполняющимся потокам информацию о том что приложение хочет остановиться. И пока ваши потоки выполняются - рестарта не произойдет! Если у вас длинная операция или рекурсивная логика (как было в моем случае) - вы будете ждать вечность пока что-то не остановит ваш(и) поток(и). Вы думаете вы хитрее и разнесете выполнение логики в отдельные треды? А вот и нет: приложение просто рухнет на рестарте! Видимо это связано с тем, что в контейнере оказывается две работающие сборки. Решение кстати в таком случае - перезапуск контейнера. Но мы же не можем при каждой поставке перезапускать контейнеры у заказчика, верно?



Так что же делать, спросите вы? Как нам быть, Егор? Что делать, если OnAppEnd не работает?

С этим вопросом я странствовал по миру и много думал. За время поисков я разгадал загадки Атлантиды, Бермудского треугольника, убийство Кеннеди и Тунгусского метеорита, но загадка рестарта приложения Creatio 7.17.4 на Net.Core не поддавалась решению. Приложение то не перезапускалось, то падало при рестарте. Долго ходил я в поисках ответа, пока случайно не заметил, что при рестарте приложения Quartz перестает запускать ежеминутные триггеры.



Во время попытки рестарта приложения планировщик заданий переходит в режим паузы (StandBy), определить это легко через класс AppScheduler: AppScheduler.Instance.InStandbyMode. Если true, то планировщик остановлен - это значит, что приложение пытается перезапуститься. Планировщик может останавливаться и по другим причинам, но эта - одна из них, которая может быть спасительной соломинкой в таких ситуациях.



Вы можете вставить проверку в логику вашего потока на AppScheduler.Instance.InStandbyMode == true и завершать его в этом случае. После завершения всех потоков приложение перезапустится.



Спасибо за внимание!

Нравится

Поделиться

1 комментарий

Кстати, на неткоре также есть особенность: запускаемые через FlowEngine процессы из EntityEventListener в фоновом режиме не будут идти по элементам (видимо также нету UserConnection). Приходится запускать их не в фоновом режиме.

Показать все комментарии

В IDS Borjomi открыта вакансия Разработчика Terrasoft



Локация: Москва или удаленный формат работы

 

Основные задачи:

✅ Разработка на платформе Creatio (Terrasoft)

✅ Оценка сроков выполнения

✅ Оптимизация и настройка текущего функционала

✅ Диагностика и устранение ошибок на уровне кода

✅ Консультации и обучение пользователей

✅ Кастомизация встроенных решений Sales, Marketing

 

Мы ожидаем:

✅Опыт работы в роли разработчика на платформе Creatio (Terrasoft) от 1 года

✅Знание платформы Creatio (Terrasoft)

✅Знание JavaScript

✅Знание C#

✅Желательно знание SQL, T-SQL (или опыт работы с PostgreSQL или Oracle) на уровне написания сложных запросов, представлений и процедур

✅Умение работать с чужим кодом

✅Опыт работы с Git желателен

✅Настройка интеграции c внешними сервисами

 

Контактное лицо

Анна

Mob.: 8 (906) 098 3145

E-mail: a.mosinyan@ids-borjomi.com 

 

Нравится

Поделиться

0 комментариев
Показать все комментарии

Подписаться на сообщение загрузки модуля из схемы EmailMessagePublisherPage

messages: {
			"SendListenerEmailData": {
				mode: this.Terrasoft.MessageMode.PTP,
				direction: this.Terrasoft.MessageDirectionType.PUBLISH
			},
			"EmailMessagePageLoaded": {
				mode: this.Terrasoft.MessageMode.PTP,
				direction: this.Terrasoft.MessageDirectionType.SUBSCRIBE
			},
		},
		methods: {
			init: function(){
				this.callParent(arguments);
				this.sandbox.subscribe("EmailMessagePageLoaded", this.onEmailMessagePageLoaded, this, [this.getEmailMessagePublisherModuleId()]);
			},
 
			onEmailMessagePageLoaded: function(args){
				debugger;
				if (this.$EntitySchemaName === "NrbCase") {
					Terrasoft.chain(
						function (next) {
							var esq = Ext.create("Terrasoft.EntitySchemaQuery", { rootSchemaName: "NrbConfigurator", rowCount: 1 });
							esq.addColumn("NrbMailboxForComunicationWithClient");
							esq.getEntityCollection(function (result) {
								if (!result.success || result.collection.count == 0) {
									throw new Error("Запись конфигуратора не найдена");
								}
								var getMailFromConfigurator = result.collection.first();
								var senderEmailConfigurator = getMailFromConfigurator.values.NrbMailboxForComunicationWithClient;
								next(senderEmailConfigurator);
							}, this);
						},
						function (next, senderEmailConfigurator) {
							data = {
								entitySchemaName: this.$EntitySchemaName,
								title: this.getMasterEntityParameterValue("Number"),
								recepientEmail: this.getMasterEntityParameterValue("NrbContactEmail"),
								contact: this.getMasterEntityParameterValue("Contact"),
								nrbObject: this.getMasterEntityParameterValue("NrbServiceObject"),
								senderEmail: senderEmailConfigurator
							}
							this.sandbox.publish("SendListenerEmailData", data, [this.getEmailMessagePublisherModuleId()]);
						}, this
					);
				}
			},

И опубликовать сообщение SendListenerEmailData

Переопределить схему EmailMessagePublisherPage

define("EmailMessagePublisherPage", ["EmailMessagePublisherPageResources"],
		function(resources) {
			return {
				entitySchemaName: "Activity",
				mixins: {
				},
				messages: {},
				attributes: {},
				methods: {
                  	setListenerEmailData: function(data) {
                      	this.callParent(arguments);
                      	debugger;
                        if(data.entitySchemaName == "NrbCase"){
							this.$Title = data.title;
                          	this.$Recepient = data.recepientEmail;
                          	this.$NrbObject = data.nrbObject.value || {};
                          	this.$Sender = data.senderEmail;
                          	this.$Contact = data.contact;
 
                        }
					},
                },
				diff: /**SCHEMA_DIFF*/[]/**SCHEMA_DIFF*/
			};
		});

 

Нравится

Поделиться

0 комментариев
Показать все комментарии
using System;
using System.Collections.Generic;
using Terrasoft.Core;
using Terrasoft.Core.DB;
using Terrasoft.Core.Entities;
 
namespace Norbit.Crm.EsPlus.Common
{
    /// <summary>
    /// Методы расширения для ESQ.
    /// </summary>
    public static class EsqExtensions
    {
        /// <summary>
        /// Постраничное чтение ESQ запроса.
        /// </summary>
        /// <param name="esq">Запрос.</param>
        /// <param name="userConnection">Пользовательское подключение.</param>
        /// <param name="rowCount">Количество считываемых записей за один раз(по умолчанию 1000).</param>
        /// <returns>Коллекция объектов из запроса.</returns>
        public static List<Entity> GetEntityCollectionPaginatedReading(
            this EntitySchemaQuery esq,
            UserConnection userConnection,
            EntitySchemaQueryOptions options = null,
            int rowCount = 1000)
        {
            if (esq == null)
            {
                throw new ArgumentNullException(nameof(esq));
            }
 
            if (userConnection == null)
            {
                throw new ArgumentNullException(nameof(userConnection));
            }
 
            var entityCollection = new List<Entity>();
            esq.UseOffsetFetchPaging = true;                              /// Использовать постраничное чтение.
 
            var esqOptions = options ?? new EntitySchemaQueryOptions                     /// Настройки получения коллекции.
            {
                PageableRowCount = rowCount,                             /// Количество считываемых записей.
                RowsOffset = 0,                                          /// Количество записей, которые необходимо пропустить
                PageableDirection = PageableSelectDirection.Next,
                PageableConditionValues = new Dictionary<string, object>()
            };
 
            var entities = esq.GetEntityCollection(userConnection, esqOptions);
 
            while (entities.Count > 0)
            {
                foreach (var entity in entities)
                {
                    entityCollection.Add(entity);
                }
                esqOptions.RowsOffset += esqOptions.PageableRowCount;
                esq.ResetSelectQuery();                                     /// Обнуление результата запроса.
                entities = esq.GetEntityCollection(userConnection, esqOptions);
            }
 
            return entityCollection;
        }
 
    }
}

 

Нравится

Поделиться

1 комментарий

Есть подозрение. что поможет 

PageableDirection = PageableSelectDirection.First  при первом запросе.

потом сменить на Next

Показать все комментарии

Ниже мы рассмотрим пример настройки процесса автоматической отправки письма с вложением с помощью бизнес-процесса. 

Для этого нам понадобятся два элемента процесса: “Обработать файл” и “Отправить Email”. 

Изображение удалено.Изображение удалено.

С помощью элемента процесса «Обработать файл» мы можем вычитать из детали «Файлы и ссылки» раздела необходимые файлы. Либо же получить этот файл из параметра процесса или сгенерировать отчет. 

В рамках этого элемента мы можем вычитанный файл использовать далее в процессе либо сохранить на деталь «Файлы и ссылки» другого раздела. 

В элементе процесса «Отправить Email», помимо стандартных настроек (От кого – Кому, и т.д.), нас интересует поле «Добавить вложение». 

Изображение удалено.



В данное поле мы и будем передавать полученный в элементе процесса «Обработать файл» файл в письмо. 

Перейдем к самому процессу. Например, нам необходимо отправить email с файлом, который находится в детали «Файлы и ссылки» из записи конкретного контакта.  

(о том как запустить процесс из раздела подробно тут

В элементе процесса «Обработать файл» мы указываем источник, откуда мы получаем файл. В нашем примере это деталь «Файлы и ссылки» объекта. 

После указываем этот самый объект: Файл и ссылка контакта, в нашем случае. 

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

После выбираем какие действия мы будем выполнять далее с  отфильтрованным файлом: используем далее в процессе (наш вариант) или сохраняем на деталь «Файлы и ссылки» другого раздела. 

Изображение удалено.

Далее, в элементе «Отправить email» мы настраиваем все необходимые параметры (от кого, кому, письмо по шаблону или произвольное и т.д.). После чего в поле «Добавить вложение» мы можем указать файл вычитанный в элементе «Обработать файл». 

Изображение удалено.

Изображение удалено.

Во вложении архив с описанным процессом.

Прикрепленные файлы

Нравится

Поделиться

0 комментариев
Показать все комментарии