https://github.com/shorvath/MammalianMethylationConsortium
Raw File
Tip revision: fb52b26a0c75f6e5d508d2d074301b1ef00036d7 authored by ahaghani on 12 November 2021, 17:56:47 UTC
Delete a
Tip revision: fb52b26
scoreProbes.py
from __future__ import print_function
import sys
import gzip
from collections import defaultdict
from itertools import combinations

def getMaxSpecies(positionComb, maxSpeciesSubset, curChunk, startPos, endPos, refSpecies, pt = False):
    # Returns the maximal subset of species that can be covered with a probe starting at startPos and ending at endPos (end exclusive)
    # with SNVs at positionComb. Also returns the specific nucleotide choice we should design for at every SNV so that
    # the maximum number of species are covered.

    # First keep only the species where the CG is conserved. The CG is always at positions startPos + 49 and startPos + 50
    finalSNVchoice = []
    localMaxSpeciesSubset = set(maxSpeciesSubset)
    for pos in range(startPos, endPos):
        SNVspecies = defaultdict(lambda: defaultdict(list))  # for each possible SNV nucleotide, which species we can cover

        maxSpeciesSubsetCopy = set(localMaxSpeciesSubset)
        for species in maxSpeciesSubsetCopy:
            if (curChunk[pos][species] != curChunk[pos][refSpecies]):  # if we have an SNV in this species
                if (curChunk[pos][species] == "X"):
                    localMaxSpeciesSubset.remove(species) # can't cover if nothing in alignment here
                else:
                    if pos in positionComb:  # check if we have an SNV here
                        SNVspecies[pos][curChunk[pos][species]].append(species) # add to the list of species we can keep if we design for this nucleotide
                    else:
                        localMaxSpeciesSubset.remove(species) # can't cover if no SNP at this position and different
        # find the best SNV choice at this position if we need one
        if len(SNVspecies[pos]) > 0:
            SNVchoices = []
            for i in SNVspecies[pos]:
                SNVchoices.append((len(SNVspecies[pos][i]), i))
            SNVchoices = sorted(SNVchoices, reverse = True)

            count = positionComb.count(pos) # how many SNVs are we picking at this position
            SNVchoices = [x[1] for x in SNVchoices[:count]]
            SNVpurpose = defaultdict(list)
            # remove the species that don't have this SNV at this position
            for i in SNVspecies[pos]:
                if i not in SNVchoices:
                    for species in SNVspecies[pos][i]:
                        localMaxSpeciesSubset.remove(species)
                else:
                    for species in SNVspecies[pos][i]:
                        SNVpurpose[i].append(species)

            for i in SNVchoices:
                finalSNVchoice.append((pos, i, curChunk[pos][refSpecies], SNVpurpose[i]))
        #if pt: print(maxSpeciesSubset)
    return localMaxSpeciesSubset, finalSNVchoice

def getFeasibleSpecies(curChunk, startPos, endPos, refSpecies, numSNV):
    # Return the feasible species for a 100bp chunk centered aroung the CG based on how many species actually
    # have the CG conserved, since we can't design for an SNV in there. Also returns the list of positions where we
    # need an SNV. If multiple SNVs are needed at a position, that position is featured multiple times in the list.
    maxSpeciesSubset = set()
    for species in curChunk[min(curChunk.keys()) + 50].keys():
        if curChunk[min(curChunk.keys()) + 50][species] + curChunk[min(curChunk.keys()) + 51][species] == "CG":
            maxSpeciesSubset.add(species)

    SNVPossibilities = []
    maxSpeciesSubsetCopy = set(maxSpeciesSubset)
    different = defaultdict(set)
    for species in maxSpeciesSubset:
        mismatches = []
        for pos in range(startPos, endPos):
            if curChunk[pos][species] != curChunk[pos][refSpecies]:
                mismatches.append((pos, curChunk[pos][species]))
                if (len(mismatches) > numSNV):
                    break
        if len(mismatches) <= numSNV:
            for i in mismatches:
                different[i[0]].add(i[1])
        else:
            maxSpeciesSubsetCopy.remove(species)

    for pos in range(startPos, endPos):
        SNVPossibilities += ([pos] * len(different[pos]))
    maxSpeciesSubset = set(maxSpeciesSubsetCopy)

    return (maxSpeciesSubset, SNVPossibilities)

def scoreProbe(probeStrand, probeType, numSNV, curChunk, infiniumType):
    # print(probeStrand, probeType, infiniumType)
    # For each probe, go through the 4 options for start and end and get the largest set of species we can
    if (probeStrand == "TOP"):
        if (probeType == "O"):
            if (infiniumType == 1):
                startPos = min(curChunk.keys()) + 1
            elif (infiniumType == 2):
                startPos = min(curChunk.keys())
            else:
                print("Wrong infinium probe type. Must be 1 or 2.")
        elif (probeType == "C"):
            if (infiniumType == 1):
                startPos = min(curChunk.keys()) + 50
            elif (infiniumType == 2):
                startPos = min(curChunk.keys()) + 51
            else:
                print("Wrong infinium probe type. Must be 1 or 2.")
        else:
            print("Wrong probe type found in column 7.")
            exit(1)
    elif (probeStrand == "BOT"):
        if (probeType == "O"):
            if (infiniumType == 1):
                startPos = min(curChunk.keys()) + 51
            elif (infiniumType == 2):
                startPos = min(curChunk.keys()) + 52
            else:
                print("Wrong infinium probe type. Must be 1 or 2.")
        elif (probeType == "C"):
            if (infiniumType == 1):
                startPos = min(curChunk.keys()) + 2
            elif (infiniumType == 2):
                startPos = min(curChunk.keys()) + 1
            else:
                print("Wrong infinium probe type. Must be 1 or 2.")
        else:
            print("Wrong probe type found in column 7.")

    maxSpecies, SNVpositions = getFeasibleSpecies(curChunk, startPos, startPos + 50, "hg19", numSNV)
    probeSequence = ""
    for pos in range(startPos, startPos + 50):
        probeSequence += curChunk[pos]["hg19"]
    bestSpecies = set()
    bestSNVchoice = []

    for positionComb in combinations(SNVpositions, min(numSNV, len(SNVpositions))):
        curMaxSpecies, curSNVchoice = getMaxSpecies(positionComb, maxSpecies, curChunk, startPos, startPos + 50, "hg19", pt=True)
        #if (positionComb == (30864656, 30864659, 30864659)):
        #    print("But out here")
        #    print(curMaxSpecies)
        #    print(len(curMaxSpecies))

        if (len(curMaxSpecies) > len(bestSpecies)):
            bestSpecies = curMaxSpecies
            bestSNVchoice = curSNVchoice
    # returns the largest list of species we can cover, the best combination of SNVs and the best SNV choice
    return bestSpecies, bestSNVchoice, probeSequence, startPos + 1, startPos + 50

def main():
    if len(sys.argv) != 4:
        print("Usage: python getCGConservationScores.py <alignment features file> <probes file> <output file>")
        exit(1)
    alignmentFile = gzip.open(sys.argv[1], 'rt')
    probesFile = gzip.open(sys.argv[2], 'rt')
    ofile = gzip.open(sys.argv[3], 'wt')

    isMammal = {"hg19": 1,"panTro4": 1,"gorGor3": 1,"ponAbe2": 1,"nomLeu3": 1,"rheMac3": 1,"macFas5": 1,"papHam1": 1,"chlSab1": 1,"calJac3": 1,"saiBol1": 1,"otoGar3": 1,"tupChi1": 1,"speTri2": 1,"jacJac1": 1,"micOch1": 1,"criGri1": 1,"mesAur1": 1,"mm10": 1,"rn5": 1,"hetGla2": 1,"cavPor3": 1,"chiLan1": 1,"octDeg1": 1,"oryCun2": 1,"ochPri3": 1,"susScr3": 1,"vicPac2": 1,"camFer1": 1,"turTru2": 1,"orcOrc1": 1,"panHod1": 1,"bosTau7": 1,"oviAri3": 1,"capHir1": 1,"equCab2": 1,"cerSim1": 1,"felCat5": 1,"canFam3": 1,"musFur1": 1,"ailMel1": 1,"odoRosDiv1": 1,"lepWed1": 1,"pteAle1": 1,"pteVam1": 1,"myoDav1": 1,"myoLuc2": 1,"eptFus1": 1,"eriEur2": 1,"sorAra2": 1,"conCri1": 1,"loxAfr3": 1,"eleEdw1": 1,"triMan1": 1,"chrAsi1": 1,"echTel2": 1,"oryAfe1": 1,"dasNov3": 1,"monDom5": 1,"sarHar1": 1,"macEug2": 1,"ornAna1": 1, "falChe1": 0,"falPer1": 0,"ficAlb2": 0,"zonAlb1": 0,"geoFor1": 0,"taeGut2": 0,"pseHum1": 0,"melUnd1": 0,"amaVit1": 0,"araMac1": 0,"colLiv1": 0,"anaPla1": 0,"galGal4": 0,"melGal1": 0,"allMis1": 0,"cheMyd1": 0,"chrPic1": 0,"pelSin1": 0,"apaSpi1": 0,"anoCar2": 0,"xenTro7": 0,"latCha1": 0,"tetNig2": 0,"fr3": 0,"takFla1": 0,"oreNil2": 0,"neoBri1": 0,"hapBur1": 0,"mayZeb1": 0,"punNye1": 0,"oryLat2": 0,"xipMac1": 0,"gasAcu1": 0,"gadMor1": 0,"danRer7": 0,"astMex1": 0,"lepOcu1": 0,"petMar2": 0}

    # Read first 100 bases from alignment file
    refSpecies = "hg19"
    curChunk = defaultdict() # dictionary of dictionaries of the nucleotide at each position for each species

    headerLine = alignmentFile.readline().rstrip().split(",")
    for i in range(102):
        splitLine = alignmentFile.readline().rstrip().split(",")
        curPos = int(splitLine[0])
        curChunk[curPos] = {}

        for j in range(2, len(splitLine)):
            if (isMammal[headerLine[j]]):
                curChunk[curPos][headerLine[j]] = splitLine[j]

    end = False
    count = 0
    for line in probesFile:
        count += 1
        if (count % 10000 == 0):
            print(count)
        splitLine = line.strip().split()
        #print(splitLine)
        coord = int(splitLine[4]) - 1
        designScore = float(splitLine[22])
        tempStrand = splitLine[15].split("_")[-1]
        if (tempStrand == "F"):
            probeStrand = "TOP"
        elif (tempStrand == "R"):
            probeStrand = "BOT"
        else:
            print ("Unknown probe strand found for ", line)
            exit(1)
        probeType = splitLine[17]

        # get to position in in the alignment that is centered on the current CG site
        while (curPos - 51 < coord):
            if (curPos % 10000 == 0):
                print(curPos)
            newLine = alignmentFile.readline()
            if (not newLine):
                end = True
                break # add what to do in this case
            newSplitLine = newLine.rstrip().split(",")
            curPos = int(newSplitLine[0])
            curChunk[curPos] = {}
            curChunk.pop(min(curChunk.keys()))

            for j in range(2, len(newSplitLine)):
                if (isMammal[headerLine[j]]):
                    curChunk[curPos][headerLine[j]] = newSplitLine[j]

        if (max(curChunk.keys()) - min(curChunk.keys()) == 101):
            printEL = False
            if (end):
                break
            # Infinium 1 probe
            if designScore >= 0.6:
                numSNV = 3
            elif designScore >= 0.5:
                numSNV = 2
            elif designScore >= 0.4:
                numSNV = 1
            elif designScore >= 0.3:
                numSNV = 0
            else:
                numSNV = -1

            if numSNV >= 0:
                bestSpecies, bestSNVchoice, probeSequence, probeStart, probeEnd = scoreProbe(probeStrand, probeType, numSNV, curChunk, infiniumType = 1)
                if (len(list(bestSpecies)) > 0):
                    ofile.write(line.strip() + "\t" + ",".join(list(bestSpecies)) + "\t" + str(probeStart) + "\t" + str(probeEnd) + "\t" + probeSequence)
                    if (len(bestSNVchoice) != 0):
                        # write position of SNPs, reference nucleotide and alternate nucleotide
                        ofile.write("\t" + ",".join([str(x[0] + 1) for x in bestSNVchoice]) + "\t" + ",".join([str(x[2]) for x in bestSNVchoice]) + "\t" + ",".join([str(x[1]) for x in bestSNVchoice]))
                        # write which species they each target
                        for x in bestSNVchoice:
                            ofile.write("\t" + ",".join(x[3]))
                    else:
                        ofile.write("\t-\t-\t-")
                    for x in range(len(bestSNVchoice), 3):
                        ofile.write("\t-")
                    printEL = True
            #print(bestSpecies, bestSNVchoice)

            #Infinium 2 probe
            numCpG = int(splitLine[-4])
            if numCpG > 3:
                numSNV = -1
            else:
                if designScore >= 0.6:
                    numSNV = 3 - numCpG
                elif designScore >= 0.5:
                    numSNV = 2 - numCpG
                elif designScore >= 0.4:
                    numSNV = 1 - numCpG
                elif designScore >= 0.3:
                    numSNV = 0
                else:
                    numSNV = -1

            # print("numSNV = ", numSNV)
            if numSNV >= 0:
                bestSpecies, bestSNVchoice, probeSequence, probeStart, probeEnd = scoreProbe(probeStrand, probeType, numSNV, curChunk, infiniumType=2)
                if (len(list(bestSpecies)) > 0):
                    ofile.write("\t" + ",".join(list(bestSpecies)) + "\t" + str(probeStart) + "\t" + str(probeEnd) + "\t" + probeSequence)
                    if (len(bestSNVchoice) != 0):
                        ofile.write("\t" + ",".join([str(x[0] + 1) for x in bestSNVchoice]) + "\t" + ",".join([str(x[2]) for x in bestSNVchoice]) + "\t" + ",".join([str(x[1]) for x in bestSNVchoice]))
                        # write which species they each target
                        for x in bestSNVchoice:
                            ofile.write("\t" + ",".join(x[3]))
                    else:
                        ofile.write("\t-\t-\t-")
                    for x in range(len(bestSNVchoice), 3):
                        ofile.write("\t-")
                    printEL = True

            if (printEL):
                ofile.write("\n")

    alignmentFile.close()
    ofile.close()

main()

back to top