Научная статья на тему 'Минимизация рисков при разработке программных средств'

Минимизация рисков при разработке программных средств Текст научной статьи по специальности «Математика»

CC BY
180
47
i Надоели баннеры? Вы всегда можете отключить рекламу.
Область наук
Ключевые слова
гибкая методология разработки / качество программных средств / минимизация рисков / гибкое управление проектами / agile software development / software quality / risks minimizing / agile project management

Аннотация научной статьи по математике, автор научной работы — Бахтизин В. В., Кузиков А. А.

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

i Надоели баннеры? Вы всегда можете отключить рекламу.
iНе можете найти то, что вам нужно? Попробуйте сервис подбора литературы.
i Надоели баннеры? Вы всегда можете отключить рекламу.

SOFTWARE DEVELOPMENT RISKS MANAGEMENT

The article examines the obstacles that are usual for software development teams working with the Scrum methodology and the impact on software quality. The authors propose a method aimed at risks minimizing when risks are connected with ineffective productivity time of Scrum-team members during Scrum iteration. The paper suggests the measures that help in the risks quantitative assessment for late performances when the risks are linked with team members’ idle time and non-optimal decomposition of scheduled work. There are algorithms to apply the method on planning and implementation stages of the Scrum development process.

Текст научной работы на тему «Минимизация рисков при разработке программных средств»

УДК 004.413

МИНИМИЗАЦИЯ РИСКОВ ПРИ РАЗРАБОТКЕ ПРОГРАММНЫХ СРЕДСТВ

В.В. Бахтизин, к.т.н., доцент; А.А. Кузиков, ж.т.н., аспирант (Белорусский государственный университет информатики и радиоэлектроники, ул. П. Бровки, 6, г. Минск, 220013, Республика Беларусь, [email protected]; [email protected])

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

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

SOFTWARE DEVELOPMENT RISKS MANAGEMENT Bakhtizin V. V., Ph.D., associate professor; Kuzikov A A, MscIT, postgraduate (Belarusian State University of Information Science and Radioelectronics, P. Brovki St., 6, Minsk, 220013, Belarus, [email protected], [email protected]) Abstract. The article examines the obstacles that are usual for software development teams working with the Scrum methodology and the impact on software quality. The authors propose a method aimed at risks minimizing when risks are connected with ineffective productivity time of Scrum-team members during Scrum iteration. The paper suggests the measures that help in the risks quantitative assessment for late performances when the risks are linked with team members' idle time and non-optimal decomposition of scheduled work.

There are algorithms to apply the method on planning and implementation stages of the Scrum development process. Keywords: agile software development, software quality, risks minimizing, agile project management.

Многие компании сегодня рассматривают возможность перехода к инкрементной и эволюционной моделям процесса разработки программных средств (ПС), так как каскадная модель процесса разработки ПС имеет ряд недостатков [1, 2]. Более того, применение гибких методологий обусловлено необходимостью выпуска качественных ПС раньше конкурентов. Гибкие методологии разработки ПС, детализирующие и расширяющие инкрементную модель, а также хорошо зарекомендовавшие себя в условиях меняющихся требований, адаптированы и достаточно широко применяются в реальных проектах.

Среди многообразия существующих гибких методологий одно из лидирующих мест занимает методология Scrum, которая, предоставляя каркас организации и управления процессом разработки ПС, успешно сочетается с методологиями, ориентированными на различные инженерные практики [3, 4], например, экстремальное программирование и разработку, управляемую тестами. Методология Scrum благодаря своей относительной простоте и адаптируемости широко используется компаниями, предоставляющими услуги по разработке и тестированию ПС.

Как правило, Scrum-команда выполняет разработку ПС по требованиям заказчика, представленным в журнале требований к продукту в виде историй в течение итераций продолжительностью от двух до шести недель [3-5]. Фиксированные временные рамки обусловливают выполнение только полезных и важных заказчику требований и вы-

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

Тем не менее Scrum-команды нередко сталкиваются с проблемами невыполнения работ, запланированных в рамках итерации, вследствие изменения цели итерации, вызванного непредвиденной сменой приоритетов пользовательских требований, и срыва графиков выполнения запланированных работ [5]. Как правило, риски, связанные с изменением цели итерации, предусмотреть сложно. Однако оценить риски невыполнения запланированных работ в рамках итерации и управлять ими вполне возможно.

Проблема срыва графиков выполнения работ в течение итерации

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

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

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

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

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

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

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

Термины и определения

Определение 1. Множество членов Scrum-команды можно описать следующим образом: М = {т1 11 = 1, Ь}, где ш^ - 1-й член команды; Ь -количество членов Scrum-команды.

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

в ПС дефекта: £ = | к = 1, К}, где sk - к-я история итерации; К - число историй, выбранных для реализации в рамках итерации.

Определение 3. Задача - независимый участок работы, выполнение которой можно поручить отдельному члену команды. Каждую историю sk можно декомпозировать на следующее множество

задач: Жк = к 11 / = 1, Ык}, где wk,i - /-я задача к-й

истории итерации; N - число задач в к-й истории.

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

Определение 4. Множество исполнителей задач к-й истории можно описать следующим образом: Ак = {ак 111 = 1, Ык}, где ак,, - /-й исполнитель, назначенный на выполнение /-й задачи к-й истории итерации; N - число исполнителей задач в к-й истории.

Каждый член команды ш^ может быть неоднократно задействован в выполнении различных задач истории то есть играть роль исполнителя одной и более задач к-й истории, либо не иметь назначенных задач, что выражается следующим образом: Assgn^.Aк^M. (1)

В общем случае имеет значение последовательность выполнения задач, так как существует зависимость между ними. Поскольку множество задач Wk составляет декомпозицию истории но к-я история считается выполненной только после приемки у владельца истории (обычно владельца продукта), можно полагать, что некоторая конечная задача представляющая собой работы по приемке истории будет зависеть от выполнения всего множества задач к-й истории. Исполнителем задачи wкfi, как правило, является лицо, выполняющее приемку истории, например владелец продукта ак,0. Следовательно, с учетом конечной задачи множеству задач Wku{wk)0} истории sк будет соответствовать множество исполнителей Ак^>{ак, о}.

Определение 5. Ориентированный граф Gк=(Vк, Ек) является представлением истории если множество его вершин Vk=Wku{wk)0} есть множество задач истории sk. Элементами множества Ек являются упорядоченные пары задач вида (ж^,-, ж^), /V/, причем задача зависит от выполнения задачи

Определение 6. Множество W'k называется множеством начальных задач, если выполнение таких задач не зависит от выполнения каких-либо других задач к-й истории, то есть

К = К6 [1, N ]}) « ' * /

V/ е 1, ык е!¥к,

к,I к '

Л

(> <,]) * Ек

Применительно к каждой задаче к-й истории будем полагать, что задача обладает свойством атомарности (неделимости) в такой мере, что задача может быть назначена только одному исполнителю а^,, а переход от выполнения задачи к выполнению одной из зависимых от нее задач ж^-, например, при (ж^,-, Wkj)&Ek и indeg(wkj)=1, возможен только при условии завершения исполнителем а^,- работ над задачей ж^,.

Свойство атомарности задачи исключает наличие в орграфе Gk петель и циклов, четко определяя изменение объема работ Scrum-команды над конкретной историей.

Поскольку в случае методологии Scrum объем работ для каждого члена команды выражается в единицах времени (часах, днях), количество времени, требующееся исполнителю для завершения задачи, будет отражать риски, связанные с несвоевременным выполнением запланированных работ.

Подобным показателем служит временная функция x(wk l, akJ, t), определяемая отображением

x:VkX (AkU{ak,c}) x(R+u{0})^R+u{0}, (2) где Vk - множество задач k-й истории; (Aku{ak>0}) - множество исполнителей, имеющих назначения в k-й истории; R+ - множество положительных действительных чисел.

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

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

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

Определение 7. Dki - это такое подмножество задач истории sk, что начало работ над задачей wki непосредственно зависит от результатов их выполнения:

Dk,i ={wkj- Vje[1, Nk], j^i, (whj, wkJ)eEk}. (3) Из определения 6 и формулы (3) следует, что для любой начальной задачи wkJeW k Dk>1=0.

Негативное влияние на выполнение планов итерации в фиксированные временные рамки оказывает неэффективное использование рабочего времени членами Scrum-команды в результате незапланированных простоев.

Рассмотрим время, необходимое исполнителю ak>, для завершения всех работ над задачей wk>1. Очевидно, что данная величина слагается из времени непосредственного выполнения исполнителем ak>, работ над задачей wkl, а также времени ожидания завершения работ над множеством задач Dk,i на момент времени t от начала итерации: Tc(wk,„ akj, t)=T(wk,b ak,„ t)+Tw(wKi, a^,, t). (4) Время ожидания Tw(wk,, akl, t) исполнителем ak>, начала работ над задачей wkJ можно определить как максимальное время, необходимое для завершения работ над задачами Dk,i.

Следует отметить, что две задачи, wkJ и wkj, назначенные разным членам Scrum-команды

(Assgn(aJy)^Assgn(aJy)), предполагают возможность их одновременного выполнения, если не существует ориентированного пути между и

и все задачи из множеств Dk,i и DJy выполнены.

Таким образом, если отображение (2) инъек-тивно, то есть в истории sk для каждого члена команды Ш1 имеется не более одной назначенной ему на исполнение задачи то на момент времени t от начала итерации время ожидания начала выполнения задачи исполнителем ак,-, таким, что Assgn(aJy)=шг, можно задать формулой

Т (wk,í > ак,,>х) = ■шах Тс (wk,], ак,], X). (5)

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

После преобразования графа Ок время ожидания начала выполнения некоторой задачи исполнителем ак,, на момент времени t от начала итерации также рассчитывается по формуле (5).

Алгоритм преобразования графа истории

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

Вход: орграф истории sk.

1. Выбрать очередного члена команды ш^ из множества М членов Scmm-команды.

2. Выбрать подмножество задач Оу:={^у I /е[0, для исполнителей которых выполняется условие Assgn(ak)I)=шг.

3. Если для члена команды ш^ в к-й истории имеется более одной назначенной ему для исполнения задачи, то есть | Оу |>1, перейти к шагу 4. Иначе перейти к шагу 11.

4. Выбрать очередную задачу wkJ из множества Оу. Для определенности будем полагать, что просмотр множества задач Оу для выбора очередной задачи организован в соответствии с увеличением индекса задачи / = 0, Ык . Если для задачи

то перейти к шагу 11. Иначе перейти к

шагу 5.

5. Из множества задач Оу выбрать очередную задачу wkj, для которой выполняется условие 1</<Мк. Если просмотрены все задачи wkj, перейти к шагу 4. Иначе перейти к шагу 6.

6. Если в графе Ок существует ориентированный путь через вершины и wkj, перейти к шагу 5. Иначе перейти к шагу 7.

7. Расширить множество дуг Ek графа Gk за счет новой дуги (ж^,, ж^) и рассчитать значение Тж^Т^^о, а^о, 0.

8. Заменить дугу (ж^,-, ж^) дугой (ж^, ж^) в множестве дуг Ek графа Gk и рассчитать значение Тж2:= Тж^о, а^о, 0.

9. Если Тж1< Тж2, перейти к шагу 10. Иначе перейти к шагу 5.

10. Заменить дугу (ж^, дугой (ж^,-, ж^) во множестве дуг графа Gk и перейти к шагу 5.

11. Если рассмотрено все множество М членов команды, завершить алгоритм. Иначе перейти к шагу 1.

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

Минимальное время простоя члена команды

Функция (5) характеризует потенциальную величину времени простоя члена Scrum-команды, назначенного исполнителем а^, задачи ж^,. Очевидно, что эффективная последовательность выполнения задач имеет место, если функция (5) принимает минимальное значение для члена команды шг в рамках всех К историй на момент времени / от начала итерации, что, учитывая (1) и (5), можно записать следующим образом:

(т, *) =

тш_

V*=1 ,к, vi=о, щ

Т

, *

(6)

т(№

t )>0

Рассчитанное по формуле (6) значение Т^ позволяет определить историю и незавершенную задачу, на выполнении которой следует сосредоточиться участнику шг Scrum-команды.

Потенциальная выполнимость истории

Рассмотрим функцию Т^ж^, а^, /), значение которой соответствует времени, необходимому для завершения работ над к-й историей на момент времени / от начала итерации. То есть значение функции Тс^^, а^, 0) отражает фактическую трудоемкость истории на момент начала итерации.

Определение 8. Потенциальная выполнимость к-й истории характеризуется величиной

Рк (') = 1 -

Т (> 1 —

Л)

Т - /

(7)

где - конечная задача к-й истории; ak)0 - исполнитель конечной задачи ж^; Т1 - время, характеризующее продолжительность итерации, в рамках которой запланирована история ^ /е[0, Т1) -время, прошедшее от начала итерации.

Значение функции (7) позволяет определить, можно ли теоретически завершить выполнение

работ в рамках заданной истории за время, оставшееся до завершения итерации.

История потенциально выполнима в рамках итерации, если р^ОеДОД]. Иначе, если р^ОсС, существует риск невыполнения к-й истории за время, оставшееся до завершения итерации. На практике, если для истории ^ справедливо рь(0)=0, существует риск невыполнения задач истории за время, ограниченное рамками итерации. Поэтому на этапе планирования работ рекомендуется применять критерий рk(0)е(0,1] для определения факта потенциальной выполнимости к-й истории.

Применение метода минимизации рисков при планировании работ в рамках итерации

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

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

iНе можете найти то, что вам нужно? Попробуйте сервис подбора литературы.

2. Выбрать очередную историю из множества £ историй, запланированных для реализации в течение итерации. Если рассмотрены все истории из множества перейти к шагу 15. Иначе перейти к шагу 3.

3. Если история не декомпозирована на задачи (^=0), перейти к шагу 4. Иначе перейти к шагу 5.

4. Декомпозировать историю на задачи

V := = }и,0}.

5. Выбрать задачу из множества задач Если рассмотрены все задачи истории % перейти к шагу 10. Иначе перейти к шагу 6.

6. Определить множество задач Д,ъ от выполнения которых зависит задача ,, основываясь на предварительной информации о задаче.

7. Если задаче , не назначен исполнитель из числа членов Scrum-команды, перейти к шагу 8. Иначе перейти к шагу 9.

8. Назначить исполнителя задачи , (А.«^^ ,):=шг).

9. Задать время непосредственного выполнения работ над задачей , на момент начала итерации (т^ ,, а^ ,, 0)) и перейти к шагу 5.

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

11. Рассчитать значение рю:= рь(0) с целью определения факта выполнимости истории в рамках итерации на момент ее начала.

12. Если для истории выполняется рие[0,1], перейти к шагу 2. Иначе перейти к шагу 13.

(

w

а

к л > к а

ь, "к

а

к,0 ' "к,0

13. Декомпозировать историю sk на подмножество историй U с меньшим числом задач.

14. Исключить историю sk из множества S историй, дополнить множество S историй подмножеством U: S:=S\(sk}uU и перейти к шагу 2.

15. Выбрать очередного члена команды m^ из множества M членов Scrum-команды.

16. Рассчитать значение минимального времени простоя члена команды m^ на момент начала итерации: Tw^^T^Wm^O).

17. Осуществить выбор членом команды m^ задачи Wk,,, для которой Tw(Wk,,, ak,,, 0)=7Vz* и Assgn(ak,,)=mi.

18. Если рассмотрено все множество M членов команды, завершить алгоритм. Иначе перейти к шагу 15.

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

при выполнении работ в рамках итерации

В соответствии с методологией Scrum предполагается, что на ежедневно проводимых совещаниях каждый член команды отразит изменение объема работ над назначенными ему задачами. Следовательно, ежедневная частота пересчета значений нестационарных функций (4)-(7) является рекомендуемой, например Д/=1 день или Д/=8 часов. С целью минимизации рисков, связанных с неэффективной занятостью членов команды в течение итерации, на этапе работы над историями и задачами итерации продолжительностью TI в момент времени t<TI от начала итерации необходимо выполнять следующие шаги.

1. Выбрать очередную историю sk из множества S историй, запланированных для реализации в течение итерации. Если рассмотрены все истории из множества S, перейти к шагу 7. Иначе перейти к шагу 2.

2. Выбрать задачу wk, , из множества задач Vk. Если рассмотрены все задачи истории sk, перейти к шагу 4. Иначе перейти к шагу 3.

3. Для задачи wk, , и ее исполнителя ak, , обновить значение функции (2) на текущий момент времени t от начала итерации и перейти к шагу 2.

4. Рассчитать значение pkt:=pk(t) с целью определения факта выполнимости истории sk в рамках итерации на момент времени t от начала итерации.

5. Если для истории sk выполняется pkte[0, 1), перейти к шагу 1. Иначе перейти к шагу 6.

6. Прекратить работы над задачами истории sk, исключить историю из множества S историй S:=S\{sk}, переместив ее в журнал продукта, и перейти к шагу 1.

7. Выбрать очередного члена команды m^ из множества M членов Scrum-команды.

8. Рассчитать время окончания работ TC1:=TC(wk,¡, ak, ,, t) члена команды над соответствующей текущей задачей wk, , (Assgn(ak, ,)=m1).

9. Если работы над текущей задачей завершены, то есть TCl=0, перейти к шагу 10. Иначе перейти к шагу 7.

10. Рассчитать значение минимального времени простоя члена команды m на момент времени t от начала итерации (T*Wl:=T*W(ml, t)).

11. Члену команды m осуществить переход к выполнению следующей назначенной ему задачи

„ для которой (3ke[1, K], 3/'е[1, Nk]) (Assgn(ak,,)=m¿ATw(Wk, „ a^,-, t)=T*w).

12. Если рассмотрено все множество M членов команды, перейти к шагу 13. Иначе перейти к шагу 7.

13. Если по истечении времени At, например, в момент проведения очередного ежедневного совещания, t<T, перейти к шагу 1. Иначе завершить алгоритм.

В заключение отметим, что предложенный в данной работе метод, основанный на формальной оценке трудоемкости работ, позволяет разрешить проблему неэффективной организации работ в Scrum-командах путем корректировки последовательности выполнения работ в течение итерации. На примере двух алгоритмов продемонстрировано применение метода на этапах планирования и выполнения работ процесса разработки, организованного в соответствии с методологией Scrum. Предложены метрики, которые позволяют количественно оценить простой членов команды (6), а также выполнимость (7) запланированных работ в рамках итерации. Применение предложенных алгоритмов при планировании и выполнении работ в рамках Scrum-итерации способствует выполнению всего объема запланированных работ по разработке и тестированию ПС, следовательно, позволяет обеспечить заявленный уровень качества разрабатываемого ПС.

Литература

1. Бахтизин В.В., Глухова Л.А. Технология разработки программного обеспечения: учеб. пособие. Минск: БГУИР, 2010. 267 с.

2. Comer E.R., Alternative Software Life Cycle Models, Software Engineering, Piscataway, NJ, IEEE Computer Society Press, 1997, pp. 404-414.

3. Kniberg H., Scrum and XP from the Trenches, C4Media, 2007, 140 p.

4. Schwaber K., The enterprise and scrum, Microsoft Press Redmond, WA, USA, 2007, 176 p.

5. Keith C., Agile game development with Scrum, Addison-Wesley, 2010, 340 p.

References

1. Bakhtizin V.V., Glukhova L.A., Tekhnologiya razrabotki programmnogo obespecheniya (Software engineering technology), Minsk, BSUIR, 2010.

2. Comer E.R., Software Engineering, Piscataway, NJ, IEEE Computer Society Press, 1997, pp. 404-414.

3. Kniberg H., Scrum and XP from the Trenches, C4Media, 2007.

4. Schwaber K., The enterprise and scrum, Microsoft Press Redmond, WA, USA, 2007.

5. Keith C., Agile game development with Scrum, Addison-Wesley, 2010.

i Надоели баннеры? Вы всегда можете отключить рекламу.