mishaploid / carrot-demography

Demographic analysis for carrot populations
0 stars 0 forks source link

Carrot demography

This repo contains a snakemake workflow to estimate the demographic history of carrot populations with the SMC++ software.

Reference: Coe, K., Bostan, H., Rolling, W. et al. Population genomics identifies genetic signatures of carrot domestication and improvement and uncovers the origin of high-carotenoid orange carrots. Nat. Plants 9, 1643–1658 (2023). https://doi.org/10.1038/s41477-023-01526-6

Key dependencies

Project organization

data/

Data is not uploaded to git, but includes the following files:

Key files

  1. Snakefile - This is the workhorse file for Snakemake and tracks the expected output files in addition to including some python code to format input variables.

  2. rules/ - Folder containing rules files (.smk) that are called by snakemake. For this project, rules/01-population_size_history.smk contains the code necessary to run SMC++ to estimate effective population size.

  3. config.yaml - This file can be edited to adjust file paths for input files and parameters to run SMC++

  4. submit.json - Specifies cluster configuration, log files, and CPU/mem usage for each rule - needs to be adjusted depending on HPC environment.

Setup

Installing snakemake

This project uses snakemake for workflow management. Follow instructions to install via conda. If running outside of the farm2 cluster at UC-Davis, will need to edit the submit.json file accordingly.

conda create --name carrot-demography
source activate carrot-demography
conda install -c conda-forge mamba
mamba install -c bioconda snakemake

A few data wrangling steps to get things formatted for SMC++:

  1. Format population and sample ids to create a python dictionary object
    This step allows easier formatting to paste input strings of population/samples for SMC++. The input is an excel file with ADMIXTURE results and two sheets, and the desired format is a text file with two columns for Population and Sample_ID.
import pandas as pd

# Read in ADMIXTURE results
# samp_info = pd.read_excel(r'data/Resequencing_population_assignments.xlsx', sheet_name = 'Ultralow_admixture')
samp_info = pd.read_excel(r'data/Passport_data.mar.30.21.xlsx', sheet_name = 'Admixture.10')

# select population group and sample id & convert to string
# samp_ids = samp_info[['Population', 'Sample_ID']].astype(str)
samp_ids = samp_info[['Largest Proportion', 'Sample_ID']].astype(str)
samp_ids.columns = ['Population', 'Sample_ID']

# replace spaces with '_'
samp_ids = samp_ids.replace(' ', '_', regex = True)

samp_ids.Population[samp_ids['Sample_ID'].isin(['GH7097', 'GH7098', 'GH7099', 'W4', 'GH7115'])] = 'Landrace_A_Wild'

# export formatted population and sample ids
# samp_ids.to_csv('data/sample_ids.txt', header = True, index = False, sep = '\t')

samp_ids.to_csv('data/sample_ids_admix10.txt', header = True, index = False, sep = '\t')
  1. Create a list of distinguished individuals as input for a composite likelihood
    This step randomly samples 10 distinguished individuals for each population and writes them out to a text file to reference in the smc++ cv command. Iterating over multiple distinguished individuals enables a composite likelihood for the smc model.
import pandas as pd

samp_ids = pd.read_table("data/sample_ids_admix10.txt")

dist_ind = samp_ids.groupby('Population', as_index = False).apply(pd.DataFrame.sample, n = 5, replace = False)

dist_ind.to_csv('data/distinguished_individuals_admix10.txt', header = True, index = False, sep = '\t')
  1. Reformat RepeatMasker results as a BED mask file
    The mask file indicates regions that should be excluded from analysis (e.g. highly repetitive, poorly mapped regions) and is necessary to distinguish from long runs of homozygosity. It should be BED format (tab delimited) with no headers and three columns: chromosome, start position, stop position.
import pandas as pd
import csv

mask = pd.read_csv('data/DCARv3.4.all.out.gff.gz', compression = 'gzip', skiprows = 3, sep = "\t", header = None)

# select chromosome and start/stop positions
mask = mask[[0,3,4]]

# sort by position
sorted_mask = mask.sort_values(by = [0,3,4])

# export formatted mask file in bed format
sorted_mask.to_csv('data/masked_regions.bed', header = False, index = False, sep = '\t', quoting = csv.QUOTE_NONE)

In the terminal, index the mask file:

bgzip data/masked_regions.bed
tabix -p bed data/masked_regions.bed.gz
  1. Quick fix for SMC++ vcf2smc error
    Ran into a traceback error for Chr1 of the Eastern Cultivated population. This was fixed by following the advice documented here. Recoded missing data using the following command:
plink --allow-extra-chr --keep-allele-order --out data/kevinG.d3.recode --recode vcf-iid --vcf data/kevinG.d3.vcf.gz