Skip to content

Karimi-Lab/inucs

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

iNucs: Inter-Nucleosome Interactions

Given nucleosome genomic coordinates, ligation junctions (in pairs format) produced by the Pairtools program, the inucs command line tool identifies interactions falling into different nucleosomes and counts them. Here, we discuss the following aspects of inucs program:

  1. Installation
  2. Usage
  3. Algorithms
  4. Publication

1 Installation

  1. Install Anaconda with Python 3.9 or above.
  2. Install the inucs dependencies using Anaconda. In a terminal where anaconda is activated type:
conda install pandas bokeh

Then, the fully self-contained python script, inucs.py, can be directly executed; for example:

python inucs.py --help

Or, if you give the script execution permission via chmod u+x inucs.py, then simply:

./inucs.py --help

2 Usage

iNucs Pipeline

As depicted in the figure above, running inucs involves two main steps: prepare and plot. There is a built-in help provided for each stage of the program, which are accessible via command line flags -h or --help.

For overall help, use the command:

./inucs.py --help

which outputs:

usage: inucs.py [-h] [-q] {prepare,plot} ...

(omitted for brevity...)

The curly brackets {} denote that either of prepare or plot can be used as a command to the program. The -h flag is to print help message, and -q suppresses the program progress output.

NOTE
The inucs program is still under development and its command line interface (CLI) may change in the future without notice.

2.1 The prepare Command

In this step, a potentially large interaction pairs file is broken into smaller pieces and the corresponding nucleosome-nucleosome interaction matrices are built.

To access the built-in help for prepare command, issue:

./inucs.py prepare --help

which outputs:

usage: inucs.py prepare [-h] [-d <working_dir>] [-n <distance>] [-N] [-C] [-z] [--refresh] <chroms_file> <nucs_file> <interas_file>

(omitted for brevity...)
  • Input

    All input files can optionally be compressed using the gzip format, in which case they need to use .gz file extension.

    • <chroms>: a file listing the chromosomes of interest

    • <nucs>: a file containing nucleosome genomic coordinates

    • <interacts>: a file containing interaction pairs produced by the Pairtools program

  • Output

    • <working_dir>: a folder containing all the intermediary and cached files. These include the resulting matrices with nucleosome-nucleosome interaction counts. The <working_dir> location may be specified using the optional arguments -d or --dir, and if it is unspecified, the program will auto-generate a name based on the name of the interactions input file, <interacts>.
  • Optional Arguments:

    • -n <distance> or --norm <distance> The normalized value for each nucleosome pair (NP) is calculated as the observed interaction count for that NP divided by the expected count. The expected count for an NP is the average of interaction count between nucleosome pairs whose distance is within the NP's average distance +/- a given <distance>. The default value for the parameter <distance> is 200.
    • -N or --no-nucs Avoid saving extra columns for the two interacting nucleosomes (chrom1 start1 end1 chrom2 start2 end2) in the generated nucleosome-nucleosome interaction matrices. Instead, it will count on nuc_id1 and nuc_id2 (which are generated automatically) to identify nucleosomes. This option reduces the size of the produced matrix files, which can improve efficiency in memory and storage usage and speed.
    • -C or --no-cache While computing the nucleosome-nucleosome interaction matrices, using the prepare command, there are many intermediate files that are stored in the <working_dir>/cache subdirectory and can optionally be removed at the end. Keeping these intermediate files can speed up the subsequent executions of the prepare command (for example with different --norm parameters). The cache files might also prove useful for some downstream analysis by the users. However, keeping cache files increases the amount of disk space requirements significantly. The option --no-cache instructs the prepare command to remove the cache files as soon as the matrix files are generated. The cache files are not needed for the subsequent executions of the plot command and can be deleted manually if desired.
    • -m or --multiprocessing Maximum number of processes (one per CPU core) that simultaneously analyse the data. Typically, this depends on the number of CPU cores and the amount of memory available on the system. For very large input files, the recommended number of processes is one per 5 GB of RAM available (e.g., use -m 12 on a system with 60 GB of free RAM). The default value is 0 (zero), which means the program attempts to guess a reasonable number of processes.
    • -z or --zip Compress the intermediary files in the <working_dir> which will be compressed using the gzip format. Please note that while this option helps to save space, it can slow down the program noticeably.
    • --refresh It starts from scratch and recreates the intermediary files. Note that refreshing may take a long time to complete, depending on the size of the input data and your hardware specifications.

2.2 The plot Command

The results produced by the prepare command are stored in the <working_dir> folder, from which the plot command can produce heatmap plots of nucleosome-nucleosome interaction counts matrix. The user specifies the DNA region of interest, and the program finds out nucleosomes within that region, and selects the submatrix nucleosome interactions within the user-specified DNA region. The final output of the plot command is a Bokeh interactive plot in html format, which can be opened in any standard browser such as Google Chrome.

To access the built-in help for plot command, use:

./inucs.py plot --help

which outputs:

usage: inucs.py plot [-h] [-s] [-p <outfile_prefix>] <working_dir> <chrom> <start_region> <end_region>

(omitted for brevity...)
  • Input

    • <working_dir> which is created using the prepare command (see above)
    • <chrom> such as III or chromX
    • <start_region> and <end_region> which specify the beginning and end of the region of interest, such as 50000 60000, within<chrom>
  • Output

    • The resulting heatmap plot file is saved inside the <working_dir> and starts with the prefix plot or <outfile_prefix> if that is specified by the user (see -p option below). The following pattern is used to generate the resulting plot file name:

      <working_dir>/<outfile_prefix>_<chrom>_<start_region>_<end_region>.html

      For example:

      yeast_wd/plot_I_50000_60000.html

    • Optional Arguments:

      • -s or --save Only save the output files and do not attempt to show them. Without this flag, the results are both saved and shown. This flag can be useful in scripts, for example.
      • -p <outfile_prefix> or --prefix <outfile_prefix> A prefix used for output file names generated for the selected subset of the nucleosome-nucleosome interaction matrix. The generated files using this prefix include submatrices for the selected regions and orientations and an html file containing the plots.

2.3 Examples

Example Data
You may download our example data for human and yeast from here.

As mentioned above ./inucs.py prepare --help gives the following usage message:

usage: inucs.py prepare [-h] [-d <working_dir>] [--refresh] [-z] <chroms> <nucs> <interacts>

As the first step, we can preprocess the input data using the following command:

./inucs.py prepare chromosomes.txt nucleosomes.txt interactions.txt -d yeast_wd

Depending on the size of input files and the system specifications, the prepare command may take a few minutes to many hours to complete.

Also, note that all input files may optionally be in gzip format.

The next step is to produce plots, for which the built-in help ./inucs.py plot --help gives:

usage: inucs.py plot [-h] [-p <outfile_prefix>] [-s] <working_dir> <chrom> <start_region> <end_region>

Following the prepare command above, we can get nucleosome interactions on chromosome I in the region between 10000 and 50000 by running the following command:

./inucs.py plot yeast_wd chrIII 10000 50000

docs/plot_chrIII_10000_50000

Plots are generated in html format, which can be opened in the default browser. Alternatively, you can simply use appropriate commands in terminal to open the plot file. For example, in macOS use:

open yeast_wd/plot_chrIII_10000_50000.html

Or, on Windows, you may use:

explorer.exe yeast_wd\plot_chrIII_10000_50000.html

Note that if you are using a Linux terminal in Windows using WSL, then you may need to escape the backslash characters, i.e., use \\ instead of \.


3 Algorithms

Given that the input data for inucs is expected to be very large, it is quite important to take the required time and space of the underlying algorithms very carefully. For the sake of brevity, here we focus only on the algorithms used in the most important parts of inucs, which make the program both efficient and scalable, allowing it to deal with very larger datasets.

3.1 Input Data Structure

The prepare command, which does all the heavy lifting for inucs, takes in three input file. We skip discussing the first input file containing chromosomes for its simplicity. The second input file contains the nucleosomes with three columns: [chrom, start, end], as in the example table below with one added column, nuc_id as an integer id for each nucleosome:

nuc_id chrom start end
1 ch1 49951 50091
2 ch1 50161 50301
3 ch1 50331 50471
...

The third input of inucs is ligation junctions file (optionally in pairs format from the Pairtools program). The following example table shows the important interaction pairs columns that are used by inucs. All other columns are quietly ignored.

... chrom1 pos1 chrom2 pos2 strand1 strand2 ...
ch1 49971 ch1 49998 - -
ch1 50182 ch1 50446 - +
...

As discussed in the Scalability section below, in order to reduce the amount of data in RAM at any given time, we first break down the interaction pairs input file into smaller chunks based on chromosomes and four orientations (strands ++, --, +-, -+). That means, when handling any given chunk of data, we will be dealing with only one chromosome (i.e., chrom1==chrom2), and one orientation. This means that we can further ignore the strands columns above and only work the following columns instead, with one added column, interact_id as an integer id for each interaction pair:

interact_id chrom1 pos1 chrom2 pos2
1 ch1 49971 ch1 49998
2 ch1 50182 ch1 50446
...

In the next sections, we will discuss these tables further and make use of the example data above.

3.2 High-level Solution Steps

Our ultimate goal is to come up with a square matrix with the size of nucleosomes that specifies how many interaction pairs exist between any given two nucleosomes. Some nucleosome input files (e.g., for humans) may contain over 10 million nucleosomes. To store the number of interaction pairs between any two nucleosomes, we need a count matrix with the size of 10,000,000*10,000,000 or 1014. This requires hundred's of tera bytes of storage! Luckily, however, most entries in the count matrix are zero and there is no interactions between most of nucleosomes. Thus, we can only store the counts that are more than zero in a sparse matrix, and consider any unspecified entry to be zero. The example table in Final Step below shows such a sparse matrix. In order to come up with that sparse matrix, we need to take one more intermediary step.

Intermediary Step:

This step starts from the last table above, and determines which side of the interaction pairs falls into which nucleosome. For our running example here, we will have:

interact_id chrom1 pos1 chrom2 pos2 nuc_id1 nuc_id2
1 ch1 49971 ch1 49998 1 2
2 ch1 50182 ch1 50446 2 3
...

Note that the last two columns, nuc_id1 and nuc_id2, have been added, with the nuc_id values coming from the nucleosomes table above. For example, when chrom2 is ch1 and pos2 is 49998, nuc_id2 fall into the second nucleosome for which id is 2.

It turns out that carrying out this intermediary step is the most challenging step computationally. We will discuss this step in more detail below.

Final Step:

Next, we can go through the last table above and simply count how many times each pair of nuc_id1 and nuc_id2 has repeated.

nuc_id1 nuc_id2 count
1 2 1
2 3 1
...

The table above is an example of the final count matrix that we were after. If we have this count matrix, then we can select any parts of it to plot a heatmap.

3.3 Algorithm Explanation

Here, we discuss the most challenging computational part of our algorithm, i.e. the intermediary step above.

To summarize the above example, we have the following two tables:

nucleosomes (with the number of rows equal to q)

nuc_id chrom start end

interaction pairs (with the number of rows equal to p)

interact_id chrom1 pos1 chrom2 pos2

And our goal is to generate the following intermediary table (from which we can compute the final counts matrix):

interact_id chrom1 pos1 chrom2 pos2 nuc_id1 nuc_id2

The number of rows for this intermediary table will be the same as interaction pairs, p.

For convenience, we define n to be the larger value of p and q, so we can say n = max(q, p). This will allow us to replace the sizes for all the three tables with n which will be an upper limit (the size the any table cannot be larger than n).

We will discuss two approaches to achieve the intermediary table above.

Search-based Algorithm (bad idea!)

One naïve approach to compute the intermediary table above is to search for the corresponding nucleosome for each side of a given interaction pair (side1: chrom1 and pos1; side2: chrom2 and pos2). That is, we can go through all the nucleosomes one by one and look for the nucleosomes matching each of the two sides of an interaction pair. This will take time to go through all nucleosomes, and we have to do that times, once for each interaction pair. Thus, in total, it will take quadratic time time for the naïve algorithm to complete. Keep in mind that n can be very large, as for example a typical human interaction pairs file may contain a few billion rows.

Example 1: To get a sense of how much time this may take in practice, let us assume that each step for the algorithm above takes one millisecond and that there are one billion interaction pairs, or n=109. Thus, there will be n2=1018 steps for the naïve algorithm to complete. Let us assume that each step takes one millisecond, so in total the algorithm could take about seconds to complete. That is more than 31 million years!

To speed up the search step, we can first sort the nucleosomes, and then we can use binary search for each side of a pair. This binary search will take only time, instead of the previous linear time, Now, for each pair, we will have to do that for both sides, so 2n times in total, which means or just by the definition of the big-O notation. Note that for this approach, we will have to sort the nucleosomes first, which will take an additional steps. This will make the total required time to be + which will still be again by the definition of the big-O notation.

Example 2: Using the same assumptions as in Example 1, for one billion interaction pairs, n=109, now there will be steps for the binary search based algorithm to complete. As before, assuming each step takes one millisecond, in total the algorithm could take about seconds or about 3.5 months to complete! This is a great improvement compared to the naïve approach.

Sorting-based Algorithm (used in inucs)

The time complexity of for the binary search based algorithm discussed above seems reasonable, and it may not be too trivial to find alternative approaches with much better complexity. However, even if we cannot improve the time complexity of , are there ways to speed up each step? It turns out that that answer is positive.

That is, if we manage to reduce the problem of matching interaction pairs with nucleosomes into a sorting problem, then we can leverage the power of existing sorting algorithms, and we can utilize vectorization used by Python libraries for sorting. The reason why this can help is discussed in more detail in the following section on efficiency.

For now, we show how we can use sorting for matching interaction pairs with nucleosomes.

Algorithm Overview
Let us define a DNA location to be specified by a chromosome name and a position, e.g., chr1 and 49951. Thus, we can say each interaction pair involves two sides or two locations, which are both specified on a single row of an interaction pair table, i.e., there are two locations per row. We can rearrange this by breaking each row into two rows, putting one location per row. This operation is called stacking, which doubles the total number rows here. Next, we also append all nucleosomes as new rows to the same stacked table, but in a way to still keep it as having one location per row. To be able to do that, consider that a nucleosome has a start and an end, and we reuse the start value for its location as well. Now, we have a very long table, each row of which has exactly one location. It is easy enough to also add extra columns to keep track of the origin of each row, which can be: the first location of an interaction pair, or the second location of a pair, or a nucleosome. It is now possible to sort this long table based on location (i.e., chromosome and position). On this sorted table, rows originated from interaction pairs are between rows originated from nucleosomes. Thus, we can easily determine which nucleosome each interaction pair location falls into, and we record that information, i.e., nucleosome id, for each row. We no longer need the rows originated from nucleosomes, so we drop them. Therefore, the remaining rows are only those that are originated from interaction pairs, but now we also have recorded the nucleosome ids they fall into. Finally, we can unstack this table, putting it back to its original format with two locations per row, or one pair per row. Of course, now, each row also contains two nucleosome ids, one for each side of the interaction pair. This is exactly the intermediary table that we discussed above.

In the following, we work through an example to further clarify how the sorting-based algorithm works. Let us consider three of the example tables above, namely, nucleosomes, interaction pairs, and intermediary tables. We start with slightly reformatting the nucleosomes by adding one column: pos with its values copied from start.

nuc_id chrom pos start end
1 ch1 49951 49951 50091
2 ch1 50161 50161 50301
3 ch1 50331 50331 50471
...

Notice that two columns are underlined [chrom pos] to indicate that these two columns together hold key values for the whole table.

Now, consider the example interaction pairs table:

interact_id chrom1 pos1 chrom2 pos2
1 ch1 49971 ch1 49998
2 ch1 50182 ch1 50446
...

We would like to be able to use the same key columns [chrom pos] for this table too. However, there are two sets of [chrom pos] columns for side1 and side2 of each pair in each row. We can reformat this table by turning each row into two rows by "stacking" them on each other. Both resulting rows will have the same interaction id interact_id, so we can always "unstack" them and go back to original format as needed. Here is the stacked version of the same table:

interact_id side chrom pos
1 side1 ch1 49971
1 side2 ch1 49998
2 side1 ch1 50182
2 side2 ch1 50446
...

What used to be in [chrom1 pos1] is now indicated by side1, and [chrom2 pos2] is now indicated by side2. Please, observe that all the information captured in the original interaction pairs table is also captured in the new stacked version of it, and vise versa. That is, the two tables are equivalent.

However, now, we are using the same key columns [chrom pos] in both reformatted tables for nucleosomes and interaction pairs. This in turn allows us to "merge" these two table as follows:

interact_id side nuc_id chrom pos start end
1 ch1 49951 49951 50091
2 ch1 50161 50161 50301
3 ch1 50331 50331 50471
1 side1 ch1 49971
1 side2 ch1 49998
2 side1 ch1 50182
2 side2 ch1 50446
...

Now, we can sort on the key columns [chrom pos]. Note that this sorting will be very efficient as it will be utilizing hardware supported vectorization which will be discussed in more details in the next section. The sorted table will be as follows:

interact_id side nuc_id chrom pos start end
1 ch1 49951 49951 50091
1 side1 ch1 49971
1 side2 ch1 49998
2 ch1 50161 50161 50301
2 side1 ch1 50182
3 ch1 50331 50331 50471
2 side2 ch1 50446
...

Now, we can easily fill in the missing values in all interaction pair rows from the the nucleosome rows just above them. (The newly added values are indicated by bold italics fonts.)

interact_id side nuc_id chrom pos start end
1 ch1 49951 49951 50091
1 side1 1 ch1 49971 49951 50091
1 side2 1 ch1 49998 49951 50091
2 ch1 50161 50161 50301
2 side1 2 ch1 50182 50161 50301
3 ch1 50331 50331 50471
2 side2 3 ch1 50446 50331 50471
...

Now, the nucleosome rows are not needed anymore and can be dropped. Also, any row whose pos does not fall into the region between start and end will also be dropped (no row in the above example are dropped for this reason.) Also, because we do not need the start and end columns anymore, we can drop them too.

Thus, we get the following resulting table:

interact_id side nuc_id chrom pos
1 side1 1 ch1 49971
1 side2 1 ch1 49998
2 side1 2 ch1 50182
2 side2 3 ch1 50446
...

As alluded to earlier, we can now use the side column to "unstack" table. We can use interact_id to decide which two rows should merge into one, and use side to decide how to rename columns. That is, rows with side1 fill in [chrom1 pos1 nuc_id1] columns, and rows with side2 fill in [chrom2 pos2 nuc_id2] columns. Thus, the unstacked version of the table above will be as follows:

interact_id chrom1 pos1 chrom2 pos2 nuc_id1 nuc_id2
1 ch1 49971 ch1 49998 1 2
2 ch1 50182 ch1 50446 2 3
...

Note that this table is exactly like the interaction pairs table, with added columns for nuc_id1 and nuc_id2.

Indeed, this is the intermediary table that we were after. That is, we managed to use sorting to find out the nucleosomes for both sides of each interaction pair.

Just as a reminder, the final step from here will be relatively easy. We can just go through the table above and simply count how many times each pair of nuc_id1 and nuc_id2 has repeated.

nuc_id1 nuc_id2 count
1 2 1
2 3 1
...

This is a sparse matrix representation of the nucleosome interactions count matrix that was our main goal to achieve.

3.4 Efficiency

Exploiting Hardware Vectorization

The sorting-based algorithm discussed above is used in inucs. It relies on sorting to carry out all he heavy lifting parts of the algorithm and it has the same time complexity of (e.g., using merge sort) as the the binary search based algorithm that we discussed earlier. However, even though the time complexity is not directly improved, each step of the algorithm is performed much faster (between one to two orders of magnitude) due to use of hardware-supported vectorization, which modern CPUs support at hardware level. "Vectorization is the process of converting an algorithm from operating on a single value at a time to operating on a set of values (vector) at one time." [Intel] Such hardware-level vectorized operations are sometimes referred to SIMD (single instruction, multiple data) operations, and are significantly faster! Furthermore, Python (as with many other modern languages) takes advantage of the power of SIMD operations in important libraries such as Pandas/NumPy (e.g., see this NumPy code here).

In other words, the sorting-based algorithm used in inucs takes advantage of hardware-supported vectorization through Pandas/NumPy libraries. In addition, there has been considerable work put into sorting algorithms, e.g. merge sort has a time complexity of .

In short, both algorithms discussed above, namely, binary search based algorithm and sorting based algorithm have the same time complexity of . However, each step in the later is much faster, significantly improving the overall performance.

Example 3: In Example 2, we assumed that each step in the binary search based algorithm would take one millisecond, and for one billion interaction pairs, that algorithm could take about 9 million seconds or about 3.5 months to complete. Now, assume that the sorting based algorithm takes the same number of steps, but through use of hardware-supported vectorization, each step is performed 100 times faster. That means, instead of 9 million seconds, the sorting based algorithm will take 90,000 seconds or 25 hours. That is, vectorization has helped to reduce the total time needed from about 3.5 months to about 1 day.

Parallelization

The performance of inucs has been greatly improved using parallelization, where the algorithm breaks down, for example, the large interaction pairs input files, and process the chunks in separate hardware cores simultaneously. This means the performance of inucs can greatly utilize the power of stronger servers with higher number of cores and RAM.

Future Efficiency Improvements:

We are planning for the next version of inucs continue to improve the efficiency through different means such as:

  • Reduce the size of the tables that need to be sorted, by reusing parts of the tables are that already sorted, which can be linearly merged together.
  • Further improve the support for parallelism for higher utilization of modern multicore CPUs.
  • Make use of lower level libraries such as Cython to improve efficiency of key areas in the algorithm which are most time consuming.

3.5 Scalability

We have put a large effort to make inucs scalable as much as possible. In its correct state, inucs can handle large amounts of input data without requiring exceeding computational resources. For example, in our testing, inucs could process more than 3.2 billion human chromosome interactions (over 54 GB zipped file size or 244G unzipped), while running on a MacBook Pro laptop with 64 GB of RAM in 3 hours. (Please see below for more information.)

Scalability in inucs is achieved primarily by breaking down the data into smaller pieces in different stages of running the program. Therefore, at any given time, there is only some manageable chunk of data in memory, and the intermediary results are constantly written on storage space and read back in as needed.

Some important breakdowns of the data are as follows

  • Input: Interaction pairs between chromosome locations

    Broken into chromosomes and orientations (strands ++, --, +-, -+); e.g., for human data that would be 24 chromosomes times 4 orientations, which means 96 data files extracted from original input.

  • Nucleosome-Nucleosome Interaction Matrix

    Broken into chromosomes and orientations (e.g., 96 files similar to above)

  • Submatrix selection for plotting

    A smaller range from the above matrix is selected by specifying: chromosome beginning-of-range end-of-range

Future Scalability Improvements:

The interaction pairs input file is very large and currently we start with breaking it down to smaller chunks for each chromosomes and orientation. This does help with scalability! However, the resulting chunks are still too large requiring substantial RAM to perform steps such as table sorting as mentioned before. It is possible to further break down these chuck into yet smaller pieces, which will in turn improve scalability further allowing to deal with even larger data requiring even less RAM, for example.

3.4 Runtime Measurements

To measure the performance of inucs, we have tested it on the following two example datasets, using three different systems:

  • Laptop, HP, Windows, 8 cores (with hyperthreading), 16 GB RAM
  • Laptop, MacBook Pro, 16 cores (with hyperthreading), 64 GB RAM
  • Server, Linux, 100 cores (with hyperthreading), 500 GB RAM

Example 1: S. cerevisiae Data

The S. cerevisiae data used:

  • For <nucs>, 77,060 nucleosomes
  • For <interacts>, 24,163,427 interactions pairs
  • With a resulting nucleosome interaction matrix size of 77,060 nucs x 77,060 nucs

The prepare commands, using 6 cores with hyperthreading (or 3 physical cores), took 43 seconds to complete for this example.

The time for plot is less than 5 seconds.

Example 2: Human Data

The human data used:

  • For <nucs>, 13,811,032 nucleosomes
  • For <interacts>, 3,220,503,431 interactions pairs
  • With a resulting nucleosome interaction matrix size of 13,811,032 nucs x 13,811,032 nucs

The prepare commands, using 6 cores with hyperthreading (or 3 physical cores), took just less than 3 hours to complete for this example.

The time for plot took just under 3 minutes.

Summary of Examples

Example 1:
S. cerevisiae
Example 1:
S. cerevisiae
Example 2:
Human
Example 2:
Human
Hardware (details above) Win, 16GB Mac, 64GB Mac, 64GB Linux, 500GB
Cores used (hyperthreaded) 6 cores 6 cores 12 cores 100 cores
Nucleosomes 77,060 77,060 13,811,032 13,811,032
Interactions pairs 24,163,427 24,163,427 3,220,503,431 3,220,503,431
Nucleosome interactions 77,060 x 77,060 77,060 x 77,060 13,811,032 x 13,811,032 13,811,032 x 13,811,032
Time to prepare (hh:mm:ss) 0:03:13 0:00:43 2:59:04 1:20:34
Time to plot (hh:mm:ss) 0:00:11 0:00:05 0:02:56 0:02:17

4 Publication

Mehrdad Oveisi, Manu Shukla, Nogayhan Seymen, Masae Ohno, Yuichi Taniguchi, Sunil Nahata, Remco Loos, Ghulam J Mufti, Robin C Allshire, Stefan Dimitrov, Mohammad M Karimi, iNucs: Inter-Nucleosome Interactions, Bioinformatics, 2021; DOI: https://doi.org/10.1093/bioinformatics/btab698.

About

iNucs: Inter-Nucleosome Interactions

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published