Опубликован: 02.03.2017 | Уровень: для всех | Доступ: платный
Лекция 6:

Классические шифры

6.5.1 Теория криптоанализа шифра Виженера

(см. [1, §6.5])

Рассмотрим шифр модульного гаммирования с уравнением

b_i=(a_i+y_i) ~(\mod  n),

для которого гамма является периодической последовательностью знаков алфавита. Такая гамма обычно получается периодическим повторением некоторого ключевого слова. Например, ключевое слово KEY дает гамму KEYKEYKEY... . Рассмотрим задачу вскрытия такого шифра по тексту одной криптограммы достаточной длины.

Пусть \mu - длина ключевого слова. Обычно криптоанализ шифра Виженера проводится в два этапа. На первом этапе определяется число \mu, на втором этапе - само ключевое слово.

Для определения числа \mu применяется так называемый тест Казиски, названный в честь Ф. Казиски, применившего его в 1863 г. Тест основан на простом наблюдении того, что два одинаковых отрезка открытого текста, отстоящих друг от друга на расстоянии, кратном \mu, будут одинаково зашифрованы. В силу этого в шифртексте ищутся повторения длины, не меньшей трех и расстояния между ними. Обратим внимание на то, что случайно такие одинаковые отрезки могут появиться в тексте с достаточно малой вероятностью.

Пусть d_1, d_2,... - найденные расстояния между повторениями и d -наибольший общий делитель этих чисел. Тогда \mu должно делить d. Чем больше повторений имеет текст, тем более вероятно, что \mu совпадает с d. Для уточнения значения \mu, можно использовать так называемый индекс совпадения, введенный в практику У. Фридманом в 1920 г.

Для строки x=x_1x_2\ldots x_m длины m, составленной из букв алфавита A, индексом совпадения в x, обозначаемым I_c(x) будем называть вероятность того, что две случайно выбранные буквы из x совпадают.

Пусть A=\{a_1,\ldots,a_n\}. Будем отождествлять буквы алфавита с числами, так что

a_1\equiv 0, \ldots, a_{n-1}\equiv n-2,  a_{n}\equiv n-1.

Теорема 6.1 Индекс совпадения в x вычисляется по формуле:

I_c(x) = \frac{\sum\limits_{i=0}^{n-1} f_i(f_i-1)}{m(m-1)},

где f_i - число вхождений буквы n_i в x, i\in Z_n.

Пусть x - строка осмысленного текста (например, английского). Допустим, как и ранее, что буквы в x появляются на любом месте текста с соответствующими вероятностями p_0,..., p_{n-1} независимо друг от друга, где p_i - вероятность появления буквы i в осмысленном тексте, i\in Z_n. В такой модели открытого текста вероятность того, что две случайно выбранные буквы из x совпадают с i\in Z_n равна p_i^2, и, следовательно,

I_c(x)\approx \sum\limits_{i=0}^{n-1} p_i^2. ( 6.1)

Взяв за основу значения вероятностей p_i для открытых текстов на английском языке, получаем приближение \sum_{i=0}^{25} p_i^2\approx 0,066. Тем самым, для английских текстов x можно пользоваться следующим приближением для индекса совпадения:

I_c(x)\approx 0,066.

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

I_c(x)\approx 0,053.

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

Таблица 6.13. Индексы совпадения европейских языков
Язык Русский Алгл. Франц. Нем. Итал. Испан.
I_c(x)\approx 0,0529 0,0662 0,0778 0,0762 0,0738 0,0775

Рассуждения, использованные при выводе формулы (4), остаются, очевидно, справедливыми и в случае, когда х - результат зашифровывания некоторого открытого текста простой заменой. В этом случае вероятности p_i переставляются местами, но сумма \sum_{i=0}^{n-1}p_i^2 остается неизменной.

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

I_c(x)=\sum\limits_{i=0}^{n-1} \frac{1}{n^2} = n\cdot \frac{1}{n^2} = \frac{1}{n}.

Вернемся к вопросу об определении числа \mu.

Пусть y= y_1 y_2\dots y_n - данный шифртекст. Выпишем его с периодом \mu:

Y_1^\downarrow Y_2^\downarrow \cdots Y_\mu^\downarrow
y_1 y_2 \cdots y_\mu
y_{\mu+1} y_{\mu+2} \cdots y_{2\mu}
y_{2\mu+1} y_{2\mu+2} \cdots y_{3\mu}
\cdots \cdots \cdots \cdots

и обозначим столбцы получившейся таблицы через Y^\downarrow_1,Y^\downarrow_2,\ldots,Y^\downarrow_\mu. Если \mu - это истинная длина ключевого слова, то каждый столбец Y^\downarrow_i, i=1,\ldots,\mu, представляет собой участок открытого текста, зашифрованный простой заменой, определяемой подстановкой

\left(\begin{array}{ccccccc}0&1&2&\ldots&n-s&\ldots&n\\ s&s+1&s+2&\ldots&0&\ldots&s-1\end{array}\right) ( 6.2)

для некоторого s\in\{0,1,\ldots,n-1\} (числа берутся по модулю n).

В силу сказанного выше, (для английского языка) I_c (Y^\downarrow_i)\approx 0,066 при любом i. С другой стороны, если \mu отлично от длины ключевого слова, то столбцы Y^\downarrow_i будут более "случайными", поскольку они являются результатом зашифровывания фрагментов открытого текста некоторым многоалфавитным шифром. Тогда I_c (Y^\downarrow_i будет ближе (для английского языка) к числу 1/28 \approx 0,038.

Заметная разница значений I_c(x) для осмысленных открытых текстов и случайных последовательностей букв (для английского языка - 0,066 и 0,038, для русского языка - 0,053 и 0,030) позволяет в большинстве случаев установить точное значение \mu.

Предположим, что на первом этапе мы нашли длину ключевого слова \mu. Рассмотрим теперь вопрос о нахождении самого ключевого слова. Для его нахождения можно использовать так называемый взаимный индекс совпадения.

Пусть x=(x_1x_2\ldots x_m), y=(y_1y_2\ldots y_{m'}) - две строки букв алфавита A. Взаимным индексом совпадения x и y, обозначаемым MI_c(x,y), называется вероятность того, что случайно выбранная буква из x совпадает со случайно выбранной буквой из y.

Пусть f_0, f_1,\ldots, f_n$ и $f^1_0, f^1_1,\ldots, f^1_{n-1} - числа вхождений букв алфавита в x и y соответственно.

Взаимный индекс совпадения в x и y вычисляется по формуле

MI_c(x,y)=\frac{\sum\limits_{i=0}^{n-1} f_i\cdot f^1_i}{m\cdot m'}. ( 6.3)

Пусть k=(k_1,\ldots, k_\mu) - истинное ключевое слово. Попытаемся оценить индексы MI_c(Y^{\downarrow}_i, Y^\downarrow_j).

Для этого напомним, что Y^\downarrow_s является результатом зашифровывания фрагмента открытого текста простой заменой, определяемой подстановкой (5) при некотором s. Вероятность того, что в $Y^\downarrow_i и Y^\downarrow_j произвольная пара букв равна 0, имеет вид p_{n-s_i}\cdot p_{n-s_j} (где p_\alpha - вероятность появления буквы \alpha в открытом тексте); вероятность того, что обе буквы есть 1, равна p_{n-s_i+1}\cdot p_{n-s_j+1}, и т. д. На основании этого получаем:

MI_C(Y^\downarrow_i, Y^\downarrow_j)\approx \sum\limits_{h=0}^{n-1} p_{h-s_i}\cdot p_{h-s_j} = \sum\limits_{h=0}^{n-1} p_h\cdot p_{h+(s_i-s_j)}.

Заметим, что сумма в правой части последнего равенства зависит только от разности (s_i-s_j) ~(\mod  n), которую назовем относительным сдвигом Y^\downarrow_i и Y^\downarrow_j. Заметим также, что

\sum\limits_{j=0}^{n-1} p_j \cdot p_{j+s ~(\mod  n)} = \sum\limits_{j=0}^{n-1} p_j \cdot p_{(j-s) ~(\mod  n)}, ( 6.4)

поэтому Y^\downarrow_i и Y^\downarrow_j с относительными сдвигами s и n-s имеют одинаковые взаимные индексы совпадения. Приведем таблицу значений сумм (6.4) для английского языка:

Таблица 6.14. Взаимный индекс совпадения при сдвиге s
Сдвиг s 0 1 2 3 4 5 6
MI_c(x,y)\approx 0,066 0,039 0,032 0,034 0,044 0,033 0,036
Сдвиг s 7 8 9 10 11 12 13
MI_c(x,y)\approx 0,039 0,034 0,034 0,038 0,045 0,039 0,043

Обратим внимание на то, что ненулевые "сдвиги" дают взаимные индексы совпадения, изменяющиеся в пределах от 0,032 до 0,045, в то время как при нулевом сдвиге индекс MI_c(x,y) близок к 0,066. Это наблюдение позволяет определить величины относительных сдвигов s_i-s_j столбцов Y^\downarrow_i и Y^\downarrow_j. Для этого заметим, что при некотором значении s(i,j)\in \{0,1,\ldots, n-1\} столбец {Y^{s(i,j)_j}}^\downarrow , полученный из Y^\downarrow_j прибавлением к каждому его элементу числа s(i,j) (по модулю n), имеет нулевой относительный сдвиг с Y^\downarrow_i.

Пусть {Y^0_j}^\downarrow,{Y^1_j}^\downarrow,\ldots,{Y^{n-1}_j}^\downarrow - результаты зашифровывания Y^\downarrow_j каждой из простых замен (6.2). Несложно вычислить взаимные индексы

MI_c(Y^\downarrow_i, {Y^s_i}^\downarrow),\quad 0\leq s\leq n-1,\quad 1\leq i<j\leq \mu

(всего, таким образом, имеется C_\mu^2\cdot n значений). Для этого воспользуемся формулой, полученной из (6.3):

MI_c(Y^\downarrow_i, {Y^s_i}^\downarrow)=\frac{\sum\limits_{h=0}^{n-1} f_h\cdot f_{h-s}^1}{m\cdot m'}.

Если s равно s_i-s_j (относительному сдвигу Y^\downarrow_i и Y^\downarrow_j), то взаимный индекс впадения должен быть (для английского языка) близок к 0,066, так как относительный сдвиг Y^\downarrow_i и {Y^s_j}^\downarrow равен нулю. Если же s не равно s_i - s_j, то взаимный индекс совпадения должен колебаться в пределах 0,032-0,045.

Используя изложенный метод, мы сможем связать системой уравнений относительные сдвиги различных пар столбцов Y^\downarrow_i и Y^\downarrow_j. В результате останется 26 (для английского языка) вариантов для ключевого слова, из которых можно выбрать наиболее предпочтительный вариант (если ключевое слово является осмысленным).

Следует отметить, что предложенный метод будет эффективным для не слишком больших значений \mu. Это объясняется тем, что для хороших сближений индексов совпадения требуются тексты достаточно большой длины.

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

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

Решение. Для определения числа букв в данном ключевом слове применяется так называемый тест Казиски. Тест основан на простом наблюдении того, что два одинаковых отрезка открытого текста, отстоящих друг от друга на расстоянии, кратном \mu (количество букв в слове), будут одинаково зашифрованы. В силу этого в шифртексте ищутся повторения длины не меньшей трех, и расстояния между ними. Необходимо обратить внимание на то, что случайно такие одинаковые отрезки могут появиться в тексте с достаточно малой вероятностью.

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

В результате получаем: 35, 85, 510, их НОД = 5.

Следовательно, с определенной долей вероятности можно заключить, что длина кодового слова равна 5.

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

Y_1^\downarrow Y_2^\downarrow Y_3^\downarrow Y_4^\downarrow Y_5^\downarrow
в л ц д у
т ж б ю ц
х ъ я р р
\ldots \ldots \ldots \ldots \ldots
т ъ ъ ж р
с щ ч л

Вычислим взаимные индексы совпадения I_c(x) букв в каждом из столбцов таблицы, для достоверного установления длины ключевого слова. Для этого посчитаем частоту повторения букв в каждом столбце.

Таблица 6.15. 1 столбец (общее количество букв m=198)
Обозначение а б в г д е ё ж з и й к л
Количество 17 2 10 16 14 7 0 1 1 3 2 1 0
Обозначение м н о п р с т у ф х ц ч ш
Количество 3 4 1 0 1 16 9 14 5 5 23 0 0
Обозначение щ ъ ы ь э ю я
Количество 5 10 3 2 2 10 11

I_c(x)=\frac{\sum\limits_{i=0}^{n-1} f_i (f_i-1)}{m(m-1)} = \frac{272+2+90+240+182+42+6+2+6+12+240}{198\cdot 197} + \frac{72+182+20+20+506+20+90+6+2+2+90+110}{198\cdot 197}=0,05676.
Таблица 6.16. 2 столбец (общее количество букв m=198)
Обозначение а б в г д е ё ж з и й к л
Количество 2 2 0 7 1 0 0 4 4 5 0 3 11
Обозначение м н о п р с т у ф х ц ч ш
Количество 3 5 2 10 18 0 2 3 14 2 7 9 11
Обозначение щ ъ ы ь э ю я
Количество 9 26 2 5 14 15 2

I_c(x)=\frac{\sum\limits_{i=0}^{n-1} f_i (f_i-1)}{m(m-1)} = \frac{2+2+42+12+12+20+6+110+6+20}{198\cdot 197} + \frac{2+90+306+2+6+182+2+42+72+110+72+650+2+20+182}{198\cdot 197} + \frac{210+2}{198\cdot 197}=0,05896.
Таблица 6.17. 3 столбец (общее количество букв m=198)
Обозначение А б в г д е ё ж з и й к л
Количество 9 24 1 1 1 2 4 0 1 0 3 10 0
Обозначение М н о п р с т у ф х ц ч ш
Количество 2 6 17 1 9 1 3 19 0 1 6 14 4
Обозначение Щ ъ ы ь э ю я
Количество 1 8 4 14 23 3 6

I_c(x)=\frac{\sum\limits_{i=0}^{n-1} f_i (f_i-1)}{m(m-1)} = \frac{72+552+2+12+6+90+2+30+272}{198\cdot 197} + \frac{72+6+342+30+182+12+56+12+182+506+6+30}{198\cdot 197} = 0,0634.
Таблица 6.18. 4 столбец (общее количество букв m=198)
Обозначение А б в г д е ё ж з и й к л
Количество 0 5 8 5 13 0 9 16 0 3 9 15 2
Обозначение М н о п р с т у ф х ц ч ш
Количество 4 9 4 14 27 5 3 13 13 2 0 1 0
Обозначение Щ ъ ы ь э ю я
Количество 2 5 0 0 0 9 2

I_c(x)=\frac{\sum\limits_{i=0}^{n-1} f_i (f_i-1)}{m(m-1)} = \frac{20+56+20+156+72+240+2+12+72+12}{198\cdot 197} + \frac{182+702+20+6+156+156+2+2+20+72+2}{198\cdot 197} = 0,0581.
Таблица 6.19. 5 столбец (общее количество букв m=197)
Обозначение А б в г д е ё ж з и й к л
Количество 15 18 1 1 0 0 1 0 1 0 2 9 1
Обозначение М н о п р с т у ф х ц ч ш
Количество 1 6 11 5 5 0 8 19 0 1 6 17 0
Обозначение Щ ъ ы ь э ю я
Количество 4 4 2 13 33 4 9

I_c(x)=\frac{\sum\limits_{i=0}^{n-1} f_i (f_i-1)}{m(m-1)} = \frac{210+306+2+72+30+110+20+20+56}{197\cdot 196} + \frac{342+30+272+12+12+2+156+1056+12+72}{197\cdot 196} = 0,0723.

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

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

MI_c(x,y)=\frac{\sum\limits_{i=0}^{n-1} f_i\cdot f_i^1 }{m\cdot m'},

где f_i, f_i^1 - частота буквы i в столбцах Y_i^\downarrow и (Y_i^1)^\downarrow соответственно; m, m' - число букв в этих столбцах.

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

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

Таблица 6.20. Сдвиг 2-го столбца на 6 позиций
Обозначение А б в г д е ё ж з и й к л
Количество 26 2 5 14 15 2 2 2 0 7 1 0 0
Обозначение М н о п р с т у ф х ц ч ш
Количество 4 4 5 0 3 11 3 5 2 10 18 0 2
Обозначение Щ ъ ы ь э ю я
Количество 3 14 2 7 9 11 9

MI_c(Y_1^\downarrow,{Y_2^6}^\downarrow)= 0,05494.
Таблица 6.21. Сдвиг 3-го столбца на 3 позиции
Обозначение а б в г д е ё ж з и й к л
Количество 23 3 6 9 24 1 1 1 2 4 0 1 0
Обозначение м н о п р с т у ф х ц ч ш
Количество 3 10 0 2 6 17 1 9 1 3 19 0 1
Обозначение щ ъ ы ь э ю я
Количество 6 14 4 1 8 4 14

MI_c(Y_1^\downarrow,{Y_3^3}^\downarrow)= 0,5798.
Таблица 6.22. Сдвиг 4-го столбца на 16 позиций
Обозначение а б в г д е ё ж з и й к л
Количество 27 5 3 13 13 2 0 1 0 2 5 0 0
Обозначение м н о п р с т у ф х ц ч ш
Количество 0 9 2 0 5 8 5 13 0 9 16 0 3
Обозначение щ ъ ы ь э ю я
Количество 9 15 2 4 9 4 14

MI_c(Y_1^\downarrow,{Y_4^{16}}^\downarrow)= 0,06068.
Таблица 6.23. Сдвиг 5-го столбца на 3 позиции
Обозначение а б в г д е ё ж з и й к л
Количество 33 4 9 15 18 1 1 0 0 1 0 1 0
Обозначение м н о п р с т у ф х ц ч ш
Количество 2 9 1 1 6 11 5 5 0 8 19 0 1
Обозначение щ ъ ы ь э ю я
Количество 6 17 0 4 4 2 13

MI_c(Y_1^\downarrow,{Y_5^3}^\downarrow)= 0,106045.

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

\begin{array}{lll}
    g[1]-g[2]=6 & g[1]=g[2] + 6 & g[2]=g[1] - 6 \\
    g[1]-g[3]=3 & g[1]=g[3] + 3 & g[3]=g[1] - 3 \\
    g[1]-g[4]=16 & g[1]=g[4] + 16 & g[4]=g[1] - 16\\
    g[1]-g[5]=3 & g[1]=g[5] + 3 & g[5]=g[1] - 3
    \end{array}

Теперь только необходимо вычислить значение g[1]:

\begin{array}{lll}
    g[l]=1: быюсю & g[l]=2: вьятя & g[l]=3: гэауа\\
    g[l]=4: дюбфб & g[l]=5: еявхв & g[l]=6: ёагцг\\
    g[l]=7: жбдчд & g[l]=8: звеше & g[l]=9: игёщё\\
    g[l]=10: йджъж & g[l]=ll: кезыз & g[l]=12: лёиьи\\
    g[1]=13: мжйэй & g[l]=14: нзкюк & g[l]=15: оилял\\
    g[l]=16: пймам & g[l]=17: pкнбн & g[l]=18: "\textbf{слово}"\\
    g[l]=19: тмпгп & g[l]=20: унрдр & g[l]=21: фосес \\
    g[l]=22: xптёт & g[l]=23: цружу & g[l]=24: чсфзф \\
    g[l]=25: штхих & g[l]=26: щуцйц & g[l]=27: ъфчкч \\
    g[l]=28: ыхшлш & g[l]=29: ьцщмщ & g[l]=30: эчънъ \\
    g[l]=31: юшыоы & g[l]=32: ящьпь &
    \end{array}

Найдено одно ключевое слово "СЛОВО".

Расшифруем зашифрованный текст:

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

Ключевое слово верное, текст читается.

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

Евгений Шаров
Евгений Шаров

как начать заново проходить курс, если уже пройдено несколько лекций со сданными тестами?

Юлия Мышкина
Юлия Мышкина

Обучение с персональным тьютором осуществляется по Скайпу или посредством переписки?

Анна Ладик
Анна Ладик
Россия, А, Университет, 2012
Паулус Шеетекела
Паулус Шеетекела
Россия, ТГТУ, 2010