Source code for integron_finder.prot_db

# -*- coding: utf-8 -*-

####################################################################################
# Integron_Finder - Integron Finder aims at detecting integrons in DNA sequences   #
# by finding particular features of the integron:                                  #
#   - the attC sites                                                               #
#   - the integrase                                                                #
#   - and when possible attI site and promoters.                                   #
#                                                                                  #
# Authors: Jean Cury, Bertrand Neron, Eduardo PC Rocha                             #
# Copyright (c) 2015 - 2018  Institut Pasteur, Paris and CNRS.                     #
# See the COPYRIGHT file for details                                               #
#                                                                                  #
# integron_finder is free software: you can redistribute it and/or modify          #
# it under the terms of the GNU General Public License as published by             #
# the Free Software Foundation, either version 3 of the License, or                #
# (at your option) any later version.                                              #
#                                                                                  #
# integron_finder is distributed in the hope that it will be useful,               #
# but WITHOUT ANY WARRANTY; without even the implied warranty of                   #
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the                    #
# GNU General Public License for more details.                                     #
#                                                                                  #
# You should have received a copy of the GNU General Public License                #
# along with this program (COPYING file).                                          #
# If not, see <http://www.gnu.org/licenses/>.                                      #
####################################################################################

from abc import ABC, abstractmethod
import os
from subprocess import call
import colorlog
from collections import namedtuple
import pandas as pd
from Bio import SeqIO, Seq
from integron_finder import IntegronError

_log = colorlog.getLogger(__name__)


"""Sequence description with fields: id strand start stop"""
SeqDesc = namedtuple('SeqDesc', ('id', 'strand', 'start', 'stop'))


[docs]class ProteinDB(ABC): """ AbstractClass defining the interface for ProteinDB. ProteinDB provide an abstraction and a way to access to proteins corresponding to the replicon/contig CDS. """
[docs] def __init__(self, replicon, cfg, prot_file=None): self.cfg = cfg self.replicon = replicon if prot_file is None: self._prot_file = self._make_protfile() else: self._prot_file = prot_file self._prot_db = self._make_db()
[docs] def __getitem__(self, prot_seq_id): """ :param str prot_seq_id: the id of a protein sequence :return: The Sequence corresponding to the prot_seq_id. :rtype: :class:`Bio.SeqRecord` object :raise KeyError: when seq_id does not match any sequence in DB """ pass
[docs] def __iter__(self): """ :return: a generator which iterate on the protein seq_id which constitute the contig. :rtype: generator """ pass
[docs] @abstractmethod def _make_protfile(self): """ Create fasta file with protein corresponding to the nucleic sequence (replicon) :return: the path of the created protein file :rtype: str """ pass
[docs] def _make_db(self): """ :return: an index of the sequence contains in protfile corresponding to the replicon """ return SeqIO.index(self._prot_file, "fasta", alphabet=Seq.IUPAC.extended_protein)
[docs] @abstractmethod def get_description(self, gene_id): """ :param str gene_id: a protein/gene identifier :return: The description of the protein corresponding to the gene_id :rtype: :class:`SeqDesc` namedtuple object :raise IntegronError: when gene_id is not a valid Gembase gene identifier :raise KeyError: if gene_id is not found in GembaseDB instance """ pass
@property def protfile(self): """ :return: The absolute path to the protein file corresponding to contig id :rtype: str """ return self._prot_file
[docs]class GembaseDB(ProteinDB): """ Implements :class:`ProteinDB` from a Gembase. Managed proteins from Proteins directory corresponding to a replicon/contig """
[docs] def __init__(self, replicon, cfg, prot_file=None): self.cfg = cfg self._gembase_path = os.path.dirname(os.path.dirname(self.cfg.replicon_path)) self.replicon = replicon # in GemBase Draft the files ar based on replicon id # but one file can contains several contig # the sequence id contains the contig number # for instance ACBA.0917.00019 vs ACBA.0917.00019.0001 # the filenames are based on replicon id # but in code the replicon id contains the contig number # for gembase complete both filename and seq_id contains contig number self._replicon_name = os.path.splitext(os.path.basename(self.cfg.replicon_path))[0] self._info = self._parse_lst() if prot_file is None: self._prot_file = self._make_protfile() else: self._prot_file = prot_file self._prot_db = self._make_db()
[docs] def _make_protfile(self): """ Create fasta file with protein corresponding to this sequence, from the corresponding Gembase protfile This step is necessary because in Gembase Draft One nucleic file can contains several contigs, but all proteins are in the same file. :return: the path of the created protein file :rtype: str """ all_prot_path = os.path.join(self._gembase_path, 'Proteins', self._replicon_name + '.prt') all_prots = SeqIO.index(all_prot_path, "fasta", alphabet=Seq.IUPAC.extended_protein) if not os.path.exists(self.cfg.tmp_dir(self.replicon.id)): os.makedirs(self.cfg.tmp_dir(self.replicon.id)) prot_file_path = os.path.join(self.cfg.tmp_dir(self.replicon.id), self.replicon.id + '.prt') with open(prot_file_path, 'w') as prot_file: for seq_id in self._info['seq_id']: try: seq = all_prots[seq_id] SeqIO.write(seq, prot_file, 'fasta') except KeyError: _log.warning('Sequence describe in LSTINFO file {} is not present in {}'.format(seq_id, all_prot_path)) return prot_file_path
[docs] @staticmethod def gembase_sniffer(lst_path): """ Detect the type of gembase :param str lst_path: the path to the LSTINFO file corresponding to the nucleic sequence :returns: either 'Complet' or 'Draft' """ with open(lst_path) as lst_file: line = lst_file.readline() if line.split()[5] in ('Valid', 'Invalid_Size', 'Pseudo'): return 'Complet' else: return 'Draft'
[docs] @staticmethod def gembase_complete_parser(lst_path, sequence_id): """ :param str lst_path: the path of of the LSTINFO file Gembase Complet :param str sequence_id: the id of the genomic sequence to analyse :return: the information related to the 'valid' CDS corresponding to the sequence_id :rtype: `class`:pandas.DataFrame` object """ dtype = {'start': 'int', 'end': 'int', 'strand': 'str', 'type': 'str', 'seq_id': 'str', 'valid': 'str', 'gene_name': 'str', 'description': 'str'} with open(lst_path) as lst_file: lst_data = [] for line in lst_file: start, end, strand, gene_type, seq_id, valid, gene_name, *description = line.strip().split() row = [start, end, strand, gene_type, seq_id, valid, gene_name, ' '.join(description)] lst_data.append(row) lst = pd.DataFrame(lst_data, columns=['start', 'end', 'strand', 'type', 'seq_id', 'valid', 'gene_name', 'description'] ) lst = lst.astype(dtype) specie, date, strain, contig = sequence_id.split('.') pattern = '{}\.{}\.{}\.\w?{}'.format(specie, date, strain, contig) genome_info = lst.loc[lst['seq_id'].str.contains(pattern, regex=True)] prots_info = genome_info.loc[(genome_info['type'] == 'CDS') & (genome_info['valid'] == 'Valid')] return prots_info
[docs] @staticmethod def gembase_draft_parser(lst_path, replicon_id): """ :param str lst_path: the path of of the LSTINFO file from a Gembase Draft :param str sequence_id: the id of the genomic sequence to analyse :return: the information related to the 'valid' CDS corresponding to the sequence_id :rtype: `class`:pandas.DataFrame` object """ lst = pd.read_csv(lst_path, header=None, names=['start', 'end', 'strand', 'type', 'seq_id', 'gene_name', 'description'], dtype={'start': 'int', 'end': 'int', 'strand': 'str', 'type': 'str', 'seq_id': 'str', 'gene_name': 'str', 'description': 'str'}, sep="\t" ) specie, date, strain, contig_gene = replicon_id.split('.') pattern = '{}\.{}\.{}\.\w?{}'.format(specie, date, strain, contig_gene) genome_info = lst.loc[lst['seq_id'].str.contains(pattern, regex=True)] prots_info = genome_info.loc[genome_info['type'] == 'CDS'] return prots_info
[docs] def _parse_lst(self): """ Parse the LSTINFO file and extract information specific to the replicon :return: """ lst_path = os.path.join(self._gembase_path, 'LSTINFO', os.path.splitext(os.path.basename(self.cfg.replicon_path))[0] + '.lst') gembase_type = self.gembase_sniffer(lst_path) if gembase_type == 'Draft': prots_info = self.gembase_draft_parser(lst_path, self.replicon.id) else: prots_info = self.gembase_complete_parser(lst_path, self.replicon.id) return prots_info
[docs] def __getitem__(self, prot_seq_id): """ :param str prot_seq_id: the id of a protein sequence :return: The Sequence corresponding to the prot_seq_id. :rtype: :class:`Bio.SeqRecord` object """ return self._prot_db[prot_seq_id]
[docs] def __iter__(self): """ :return: a generator which iterate on the protein seq_id which constitute the contig. :rtype: generator """ return (seq_id for seq_id in self._info['seq_id'])
[docs] def get_description(self, gene_id): """ :param str gene_id: a protein/gene identifier :return: The description of the protein corresponding to the gene_id :rtype: :class:`SeqDesc` namedtuple object :raise IntegronError: when gene_id is not a valid Gembase gene identifier :raise KeyError: if gene_id is not found in GembaseDB instance """ try: specie, date, strain, contig_gene = gene_id.split('.') contig_gene = contig_gene[1:] # remove the first letter b/i except ValueError: raise IntegronError("'{}' is not a valid Gembase protein identifier.".format(gene_id)) pattern = '{}\.{}\.{}\.\w?{}'.format(specie, date, strain, contig_gene) seq_info = self._info.loc[self._info['seq_id'].str.contains(pattern, regex=True)] if not seq_info.empty: return SeqDesc(seq_info.seq_id.values[0], 1 if seq_info.strand.values[0] == "D" else -1, seq_info.start.values[0], seq_info.end.values[0], ) else: raise KeyError(gene_id)
[docs]class ProdigalDB(ProteinDB): """ Creates proteins from Replicon/contig using prodigal and provide facilities to access them. """
[docs] def _make_protfile(self): """ Use `prodigal` to generate proteins corresponding to the replicon :return: the path of the created protfile :rtype: str """ if not os.path.exists(self.cfg.tmp_dir(self.replicon.id)): os.makedirs(self.cfg.tmp_dir(self.replicon.id)) prot_file_path = os.path.join(self.cfg.tmp_dir(self.replicon.id), self.replicon.id + ".prt") if not os.path.exists(prot_file_path): prodigal_cmd = "{prodigal} {meta} -i {replicon} -a {prot} -o {out} -q ".format( prodigal=self.cfg.prodigal, meta='' if len(self.replicon) > 200000 else '-p meta', replicon=self.cfg.replicon_path, prot=prot_file_path, out=os.devnull, ) try: _log.debug("run prodigal: {}".format(prodigal_cmd)) returncode = call(prodigal_cmd.split()) except Exception as err: raise RuntimeError("{0} failed : {1}".format(prodigal_cmd, err)) if returncode != 0: raise RuntimeError("{0} failed returncode = {1}".format(prodigal_cmd, returncode)) return prot_file_path
[docs] def __getitem__(self, prot_seq_id): """ :param str prot_seq_id: the id of a protein sequence :return: The Sequence corresponding to the prot_seq_id. :rtype: :class:`Bio.SeqRecord` object """ return self._prot_db[prot_seq_id]
[docs] def __iter__(self): """ :return: a generator which iterate on the protein seq_id which constitute the contig. :rtype: generator """ return (seq_id for seq_id in self._prot_db)
[docs] def get_description(self, gene_id): """ :param str gene_id: a protein/gene identifier :returns: The description of the protein corresponding to the gene_id :rtype: :class:`SeqDesc` namedtuple object :raise IntegronError: when gene_id is not a valid Gembase gene identifier :raise KeyError: if gene_id is not found in ProdigalDB instance """ seq = self[gene_id] try: id_, start, stop, strand, *_ = seq.description.split(" # ") except ValueError: raise IntegronError("'{}' is not a valid Prodigal protein identifier.".format(gene_id)) start = int(start) stop = int(stop) strand = int(strand) return SeqDesc(id_, strand, start, stop)