.. _seq-annotations: Advanced sequence handling ========================== .. sectionauthor:: Gavin Huttley Individual sequences and alignments can be manipulated by annotations. Most value in the genome sequences arises from sequence annotations regarding specific sequence feature types, e.g. genes with introns / exons, repeat sequences. These can be applied to an alignment either using data formats available from genome portals (e.g. GFF, or GenBank annotation formats) or by custom assignments. Annotations can be added in two ways: using either the ``add_annotation`` or the ``add_feature`` method. The distinction between these two is that ``add_features`` is more specialised. Features can be thought of as a type of annotation representing standard sequence properties eg introns/exons. Annotations are the more general case, such as a computed property which has, say a numerical value and a span. For illustrative purposes we define a sequence with 2 exons and grab the 1\ :sup:`st` \ exon: .. jupyter-execute:: from cogent3 import DNA s = DNA.make_seq("aagaagaagacccccaaaaaaaaaattttttttttaaaaaaaaaaaaa", name="Orig") exon1 = s.add_feature("exon", "exon1", [(10, 15)]) exon2 = s.add_feature("exon", "exon2", [(30, 40)]) Here, '``exon``' is the feature type, and '``exon#``' the feature name. The feature type is used for the display formatting, which won't be illustrated here, and also for selecting all features of the same type, shown below. We could also have created an annotation using the ``add_annotation`` method: .. jupyter-execute:: from cogent3.core.annotation import Feature s2 = DNA.make_seq("aagaagaagacccccaaaaaaaaaattttttttttaaaaaaaaaaaaa", name="Orig2") exon3 = s2.add_annotation(Feature, "exon", "exon1", [(35, 40)]) We can use the features (eg ``exon1``) to get the corresponding sequence region. .. jupyter-execute:: s[exon1] You can query annotations by type and optionally by label, receiving a list of features: .. jupyter-execute:: exons = s.get_annotations_matching("exon") print(exons) We can use this list to construct a pseudo-feature covering (or excluding) multiple features using ``get_region_covering_all``. For instance, getting all exons, .. jupyter-execute:: print(s.get_region_covering_all(exons)) s.get_region_covering_all(exons).get_slice() or not exons (the exon *shadow*): .. jupyter-execute:: print(s.get_region_covering_all(exons).get_shadow().get_slice()) The first of these essentially returns the CDS of the gene. Features are themselves sliceable: .. jupyter-execute:: exon1[0:3].get_slice()