\documentclass{article} \usepackage{amsmath} \usepackage{amssymb} \usepackage{hyperref} \title{\href{https://github.com/kaegi/aligner}{aligner}'s Algorithm} \author{kaegi} \date{\today} % Hint: \title{what ever}, \author{who care} and \date{when ever} could stand % before or after the \begin{document} command % BUT the \maketitle command MUST come AFTER the \begin{document} command! \begin{document} \maketitle \section{Notation} \label{notation} Let $n$ be the number of lines in the incorrect subtitle file. Let $n_r$ be the number of lines in the reference subtitle file. $s[0]$ to $s[n-1]$ denote the timespans for the subtitle lines in the incorrect subtitle file, $s_r[0]$ to $s_r[n_r-1]$ denote the timespans for subtitle lines in the reference subtitle file. The integer values $s[i].start$, $s[i].end$ and $s[i].length=s[i].end-s[i].start$ denote the start and end timestamps, as well as the length of every line (simlar for $s_r[i]$). The algorithm is based on discrete integer timestamps, meaning that real numbers as timestamps have to be converted into intergers in an arbitrary resolution (aligner uses milliseconds as resolution by default). The lines have to be sorted, so that $s[i].start \leq s[i+1].start$ is true for all $0\leq i s[i+1].start$. For simplicity in the formulas later on, all lines (incorrect and reference lines) are shifted by the same time so that $s_r[0].start=m$. \section{Metric} \label{metric} At the core of the algorithm is the \emph{rating} of an alignment. For each pair of subtitles/timespans $a=s_c[i]$ and $b=s_r[j]$ the rating is \[r(a,b)=\frac{\text{overlapping\_time}(a, b)}{\text{max}(a.length, b.length)}\] where $\text{overlapping\_time}$ is defined as following: \[ \text{overlapping\_time}(a,b)= \begin{cases} \text{max}(0,a.end - b.start) & \text{if } a.start \leq b.start \\ \text{max}(0,b.end - a.start) & \text{if } b.start > a.start \\ \end{cases} \] The maximum of this rating $r(s_c[i], s_r[j])$ is 1, if and only if $s_c[i].start=s_r[j].start \land s_c[i].end = s_r[j].end$, because the overlapping time can not be greater than the length of the individiual lines. In case two lines do not overlap, the rating is 0. A first attempt for the \emph{total rating of an alignment} can be: \[\text{total\_rating}=\sum_{i=0}^{n-1}{\sum_{j=0}^{n_r-1}{r(s_c[i], s_r[j])}}\] This formula will give the highest rating to an alignment, where every line $s[i]$ is moved to the most similar reference line $s_r[j]$ (in terms of the length). There are two problems with it: \begin{itemize} \item The order of the incorrect lines can be changed. A basic assumption should be that the lines might have wrong timings, but not the wrong order. \item Every line probably gets shifted by a different delta. We would like to "group" many lines which get the same delta (so only long breaks get filtered out). \end{itemize} The first point can be addressed by specifying a simple constraint. All valid alignments have to have $s_c[i].start \leq s_c[i+1].start$ for all $0\leq i 0 \text{ and } t = 0 \\ \text{max}\{p(i,t-1),p(i-1,t) + o(i,t)\} & \text{if }i > 0 \\ \end{cases} \] The value $p(n-1,2m+m_r-1)$ is the total rating where the constraint is satisfied, but still \emph{without the split-penalty}. It is easy to see that in phase $0$ (aligning no incorrect lines to the reference file), the rating of that alignment is $0$. To simplify the problem, we assume `start(sub)` is the start timestamp in milliseconds of a subtitle line `sub` and `0 <= start(sub)` is true. Let's say `get$\_$rating(t, n)` computes the best rating/alignment for the first `n` incorrect subtitle lines with the additional constraint `0 <= start(sub) <= t` for each of these `n` subtitles. Of course we can now simply set `get$\_$rating(t, 0) = 0` because if we have no incorrect subtitles to align, we have a rating of zero (independent of `t`). Now we handle the case `get$\_$rating(0, 1)`. We can simply compute the overlapping rating (where the first incorrect subtitle starts at the \''zero timepoint\'') with every reference subtitle and add up these values. With `get$\_$rating(1, 1)` things get interesting. We can either have `start(sub) = 0` or `start(sub) = 1`. Fortunaly we already have `get$\_$rating(0, 1)`, so we only need the rating where `start(sub) = 1`. This can be computed by adding up all overlapping ratings. Similarly we can compute `get$\_$rating(2, 1)` by taking the maximum of `get$\_$rating(1, 1)` and the rating where `start(sub) = 2`. In this vein we can create `get$\_$rating(t+1, 1)` from `get$\_$rating(t, 1)`. We can also speed up computing the overlapping rating, because the subtitle line will only be shifted by 1ms from `start(sub) = t` to `start(sub) = t + 1`. The subtitle will lose the rating for the segment `[t, t + 1]` and gain the overlapping rating for the segment `[t + length(sub), t + 1 + length(sub)]` on the other side. By creating a lookup-table for reference subtitles for every `t` this process has a runtime of `O(1)`. When do we stop? Well, the rating won't change anymore if `t` gets big. At the latest when `start(sub)` is be greater than any of the reference subtitle lines, because after that the overlapping rating will always be zero. Let's call `max$\_$t` the timepoint where all incorrect subtitles have been moved behind the reference subtitles. The best total rating is then `get$\_$rating(max$\_$t, number$\_$of$\_$incorrect$\_$subtitles)`. Now we have all `get$\_$rating(0, 1)` to `get$\_$rating(max$\_$t, 1)`. To compute `get$\_$rating(0, 2)`, which means that `0 <= start(sub0) <= 0` and `0 <= start(sub1) <= 0`. We already have the rating for `sub0` in form of `get$\_$rating(0, 1)`. We only need to add the overlapping rating for `sub1`. To get `get$\_$rating(1, 2)` we can either use `get$\_$rating(0, 2)` (we leave `sub1` where it is), or move `start(sub1)` to 1, which allows `start(sub0)` to be in `0 <= start(sub0) <= start(sub1) = 1`. The best rating for `sub0` for that range has been computed with `get$\_$rating(1, 1)`, we only need to add the overlapping rating for `start(sub1) == 1`. We proceed similarly to get `get$\_$rating(t+1, 2)`: leave the `sub1` like it was for `get$\_$rating(t,2)` or reposition the subtitle to `start(sub1) == t+1` and use `get$\_$rating(t+1,1) + overlapping$\_$rating(sub1,t+1)`. With the same principle we proceed with `subN`: - initialize `get$\_$rating(0, n) = get$\_$rating(0, n - 1) + overlapping$\_$rating(subN, 0)` - choose for `get$\_$rating(t+1, n)` the maximum of - `get$\_$rating(t, n)` which means "leaving `subN`" and - `get$\_$rating(t+1, n-1) + overlapping$\_$rating(t+1, subN)` which means repositioning the `subN` Until now we didn't use the `split$\_$penalty`. We need to add the split penalty when `start(subN) - start(subN+1)` is a specific value (the original distance `diff(N)`). The trick here is seeing that we only need to consider the "repostion choice". The only time `get$\_$rating($\_$, n-1)` is consulted after the inital phase is when `subN` gets repositioned. `subN` will then start at `t+1` and we consult `get$\_$rating(t+1, n-1)`. So if `subN-1` were positioned at `t+1-diff(N-1)` for `get$\_$rating(t+1-diff(N-1), n-1)` we'd be able to get the `split$\_$penalty`. This is exactly the thing we will do when we are in a phase `n`: We will not only have the "leave choice" or "reposition choice" but also the "nosplit choice". If we compute `get$\_$rating(t, n)`, we can also compare the two other values with `get$\_$rating(t-diffN, n-1) + overlapping$\_$rating(t-diffN, subN) + split$\_$penalty`. The `get$\_$rating(t-diffN, n-1) + overlapping$\_$rating(t-diffN, subN)` is again the best rating where `start(subN) = t-diffN`. We are allowed to add the `split$\_$penalty` because in the next phase `n+1`, `subN+1` will start at `t` when `get$\_$rating(t-diffN, n)` is looked up. So the final rating algorithm is: - initialize `get$\_$rating(t, 0)` with 0 - initialize `get$\_$rating(0, n) = get$\_$rating(0, n - 1) + overlapping$\_$rating(subN, 0)` - choose for `get$\_$rating(t+1, n)` the maximum of - `get$\_$rating(t, n)` which means "leaving `subN`" and - `get$\_$rating(t+1, n-1) + overlapping$\_$rating(t+1, subN)` which means repositioning the `subN` and - `get$\_$rating(t+1-diffN, n-1) + overlapping$\_$rating(t+1-diffN, subN) + split$\_$penalty` which means doing a nosplit-repositioning for `subN` To get the final alignment, we save for each phase `n` and `t+1` where `subN` was positioned (can be `t+1`, `t+1-diffN` or the previous position). If we look up that value for `n = number$\_$of$\_$incorrect$\_$subtitles` and `t = max$\_$t`, we know where the last subtitles `subN` has to be. We then know `start(subN)`. The best alignment of all previous subtitles is then computed with `get$\_$rating(start(subN), n-1)`. So we look up the position for `subN-1` in that table with `n' = n-1` and `t' = start(subN)`. That way we get all corrected positions of all incorrect subtitles and are done! Though this algorithm works (and was implemented in one of the early versions of `aligner`) it is neither fast nor space-efficient. Let's take a `45 minutes = 2700000 milliseconds < max$\_$t` subtitle file which has about `n = 900 subtitles` (these are realistic values). We build a table of `max$\_$t * n = 2430000000 ` entries. We can discard the ratings of the phase `n-1` after phase `n`, but we always need to store the positions of the subtitle `subN`. Let's assume we need 4 bytes to store them: we then have a table of `2430000000 * 4 bytes = 9720000000 bytes = 9 GB` of data in RAM!!! Even filling the table with zeros might take some noticeable time. But as it turns out we can compress that table in under 2 MB (most of the time; probably the best compression I've ever seen) with `delta encoding`. The empirical foundation is that the choices almost never change from one `t` to `t+1` (about 10 to 1000 times for one phase). If we always take the - "leave choice", the position will always be `t+1` for every `t+1` (rise by 1) - "nosplit-reposition choice", the position will always be `t+1-diffN` (rise by 1) - "reposition choice", the position won't change from `t - 1` (constant) So if we store values in a `(start, delta, length)` tuple, where the first uncompressed value is `start + 0 * delta`, the second is `start + 1 * delta`, the third is `start + 2 * delta`, ..., the last is `start + length * delta`, we can compress an entire phase into a few bytes. The same thing is applicable to the ratings. Without going into details: if we take the overlapping rating of a incorrect subtitle to a reference subtitle and "move" the incorrect subtitle from the far left to the far right we will have five segments of compressed values (first the rating will be zero, then rise linearly, then be constant, then fall linearly, then be zero again). The comparisons/choices can then be done for rating segments instead of single `t`. This yields a speedup of at least one order of magnitude. \begin{itemize} \item article \item book \item report \item letter \end{itemize} \begin{enumerate} \item article \item book \item report \item letter \end{enumerate} \begin{description} \item[article\label{article}]{Article is \ldots} \item[book\label{book}]{The book class \ldots} \item[report\label{report}]{Report gives you \ldots} \item[letter\label{letter}]{If you want to write a letter.} \end{description} \section{Conclusions}\label{conclusions} There is no longer \LaTeX{} example which was written by \cite{doe}. \begin{thebibliography}{9} \bibitem[Doe]{doe} \emph{First and last \LaTeX{} example.}, John Doe 50 B.C. \end{thebibliography} \end{document}