Changed the names of the sequences to allow for the sequencing ids that Sarah has used.
See the jupyter notebook seq_processing for the amphiprion side of this protocol. Start there.
Create an index file for the Pools that is tab separated done
# read baits table
baits <- lab %>%
tbl("baits") %>%
collect() %>%
select(baits_id, seq)
# read PCR table
pcr <- lab %>%
tbl("pcr") %>%
collect() %>%
filter(SEQ %in% params$seq) %>%
select(pcr_id, bait_id, index)
# join pcr_id and index to seq id
pools <- left_join(pcr, baits, by = c("bait_id" = "baits_id")) %>%
select(seq, pcr_id, index)
# pull in the barcodes for illumina indexes
index <- lab %>%
tbl("illumina") %>%
collect()
# join the barcodes to the seq and pcr ids
pools <- left_join(pools, index, by = c("index" = "index_num")) %>%
select(seq, pcr_id, index_code)
# create a list of the multiple seqs
seqs <- select(pools, seq) %>%
distinct()
for(i in seq(seqs$seq)){
x <- pools %>%
filter(seq == seqs$seq[i]) %>%
select(pcr_id, index_code)
# write the files for amphiprion
# readr::write_tsv(x, path = paste0("index-", seqs$seq[i], ".tsv"), col_names = F)
}
Move the files from the laptop to amphiprion using the fetch program, place the index files in the log folder for each seq ** done**
See Michelle if you need a serial number for Fetch
Create a names file for each pool done
The names have to be species, underscore and then the sample identifier, so APCL_L5432 for ligation_id L5432
# read the ligations
ligs <- lab %>%
tbl("ligation") %>%
filter(pool %in% pools$pcr_id) %>%
select(ligation_id, barcode_num, pool) %>%
collect()
# read the barcodes
barcode <- lab %>%
tbl("barcodes") %>%
collect()
# join the ligation_id and pool ids to the barcodes
ligs <- left_join(ligs, barcode, by = "barcode_num")
# join the ligs to the seq_ids
ligs <- left_join(ligs, pools, by = c("pool" = "pcr_id")) %>%
# adjust the ligation name for dDocent
mutate(name = paste0("APCL_", ligation_id)) %>%
select(seq, pool, name, barcode) %>%
# reduce the pool to only a number
mutate(pool = substr(pool, 2,5))
# for seq04, the seq column is empty. Replace it.
ligs <- ligs %>%
mutate(seq = params$seq)
# write files for amphiprion
# create a list of the multiple seqs
names <- select(ligs, pool) %>%
distinct()
# loop through all of the pools
for(i in seq(names$pool)){
x <- ligs %>%
filter(pool == names$pool[i]) %>%
select(name, barcode)
# write the files for amphiprion
readr::write_tsv(x, path = paste0("names_", names$pool[i], ".tsv"), col_names = F)
}
Move the files from the laptop to amphiprion using the fetch program, place the names files in the log folder for each seq done
Concatenate the results in the bcsplit folder - takes about a minute done
library(readr)
for (j in seqs$seq){
y <- pools %>%
filter(seq == j)
# write_file("#!/bin/bash", path = paste0(j,"_cat_all.sh"), append = F)
# add a line return
# write_file("\n", path = paste0(j,"_cat_all.sh"), append = T)
for (i in seq(y$pcr_id)){
z <- formatC(i, width = 2, flag = 0)
# generate the code
x <- paste("cat", paste0("./lane1/", y$pcr_id[i], "-read-1.fastq.gz"), paste0("./lane2/", y$pcr_id[i], "-read-1.fastq.gz"), ">", paste0("../", z, "Pool/", y$pcr_id[i], ".fastq.gz"), sep = " ")
# add the code to the file
# write_file(x, path = paste0(j,"_cat_all.sh"), append = T)
# add a line return
# write_file("\n", path = paste0(j,"_cat_all.sh"), append = T)
}
}
Move the scripts to amphiprion into the seq scripts folder with fetch
With fetch you can also highlight all of the scripts and click get info, execute instead of having to type in the chmod u+x command for each file.
write a script to generate command line to run process radtags scripts
for (j in seqs$seq){
y <- pools %>%
filter(seq == j)
write_file("#!/bin/bash", path = paste0(j,"_process_all.sh"), append = F)
# add a line return
write_file("\n", path = paste0(j,"_process_all.sh"), append = T)
for (i in y$pcr_id){
# generate the code
x <- paste("nohup", paste0("./scripts/", i, "_process.sh &"), sep = " ")
# add the code to the file
write_file(x, path = paste0(j,"_process_all.sh"), append = T)
# add a line return
write_file("\n", path = paste0(j,"_process_all.sh"), append = T)
}
}
started the last one at 9:45pm on Monday 9/24/2018 - should take about 5 hours done before 6am on Tuesday
Write text for moving files
for (j in seqs$seq){
y <- pools %>%
filter(seq == j)
write_file("#!/bin/bash", path = paste0(j,"_move_radlogs.sh"), append = F)
# add a line return
write_file("\n", path = paste0(j,"_move_radlogs.sh"), append = T)
for (i in seq(y$pcr_id)){
z <- formatC(i, width = 2, flag = 0)
# generate the code
x <- paste("mv", paste0(z, "Pool/process_radtags.log"), paste0("./logs/process", y$pcr_id[i], ".log"), sep = " ")
# add the code to the file
write_file(x, path = paste0(j,"_move_radlogs.sh"), append = T)
# add a line return
write_file("\n", path = paste0(j,"_move_radlogs.sh"), append = T)
}
}
Which samples failed?
upligs <- lab %>%
tbl("ligation") %>%
filter(pool %in% current$pcr_id) %>%
mutate(total_reads = as.numeric(total_reads),
lack_rad_tag = as.numeric(lack_rad_tag),
low_quality = as.numeric(low_quality),
retained = as.numeric(retained)) %>%
collect()
hist(upligs$total_reads, breaks = 100)
# zoom in to the lower end
zoom <- upligs %>%
filter(total_reads < 200000)
hist(zoom$total_reads, breaks = 10)
low <- upligs %>%
mutate(total_reads = as.numeric(total_reads)) %>%
filter(total_reads < 100000) %>%
select(ligation_id)
summary(upligs)
maximum <- upligs %>%
filter(retained > 1000000)
total <- upligs %>%
summarise(sum = sum(total_reads))
Rename the process radtags output to sample names
write renaming text
library(readr)
for (j in seqs$seq){
y <- pools %>%
filter(seq == j) %>%
arrange(pcr_id)
write_file("#!/bin/bash", path = paste0(j,"_all_rename.sh"), append = F)
# add a line return
write_file("\n", path = paste0(j,"_all_rename.sh"), append = T)
for (i in seq(y$pcr_id)){
z <- formatC(i, width = 2, flag = 0)
c <- substr(y$pcr_id[i], 2, 5)
# generate the code
a <- paste("cd", paste0(z, "Pool/"), sep = " ")
# add the code to the file
write_file(a, path = paste0(j,"_all_rename.sh"), append = T)
# add a line return
write_file("\n", path = paste0(j,"_all_rename.sh"), append = T)
x <- paste("sh rename.for.dDocent_se_gz", paste0("../logs/names_", c, ".tsv"), sep = " ")
# add the code to the file
write_file(x, path = paste0(j,"_all_rename.sh"), append = T)
# add a line return
write_file("\n", path = paste0(j,"_all_rename.sh"), append = T)
write_file("mv APCL* ../samples/", path = paste0(j,"_all_rename.sh"), append = T)
write_file("\n", path = paste0(j,"_all_rename.sh"), append = T)
write_file("cd ..", path = paste0(j,"_all_rename.sh"), append = T)
write_file("\n", path = paste0(j,"_all_rename.sh"), append = T)
}
}
Trim and map the reads
michelles 2018-09-25 07:25:10 samples $ dDocent dDocent 2.5.5
Contact jpuritz@uri.edu with any problems
Checking for required software /local/home/michelles/14_programs/dDocent/dDocent: line 84: [: 1.0-r82: integer expression expected
All required software is installed!
dDocent run started Tue Sep 25 07:25:15 EDT 2018
576 individuals are detected. Is this correct? Enter yes or no and press [ENTER] yes Proceeding with 576 individuals dDocent detects 40 processors available on this system. Please enter the maximum number of processors to use for this analysis. 30 dDocent detects 252 gigabytes of maximum memory available on this system. Please enter the maximum memory to use for this analysis in gigabytes For example, to limit dDocent to ten gigabytes, enter 10 This option does not work with all distributions of Linux. If runs are hanging at variant calling, enter 0 Then press [ENTER] 150
Do you want to quality trim your reads? Type yes or no and press [ENTER]? yes
Do you want to perform an assembly? Type yes or no and press [ENTER]. no
Reference contigs need to be in a file named reference.fasta
Do you want to map reads? Type yes or no and press [ENTER] yes BWA will be used to map reads. You may need to adjust -A -B and -O parameters for your taxa. Would you like to enter a new parameters now? Type yes or no and press [ENTER] yes Please enter new value for A (match score). It should be an integer. Default is 1. 1 Please enter new value for B (mismatch score). It should be an integer. Default is 4. 4 Please enter new value for O (gap penalty). It should be an integer. Default is 6. 6 Do you want to use FreeBayes to call SNPs? Please type yes or no and press [ENTER] no
Please enter your email address. dDocent will email you when it is finished running. Don’t worry; dDocent has no financial need to sell your email address to spammers. stuart620@gmail.com
At this point, all configuration information has been entered and dDocent may take several hours to run. It is recommended that you move this script to a background operation and disable terminal input and output. All data and logfiles will still be recorded. To do this: Press control and Z simultaneously Type ‘bg’ without the quotes and press enter Type ‘disown -h’ again without the quotes and press enter
Now sit back, relax, and wait for your analysis to finish
Finished at 9:47am same day
Once trimming is done, prepare to call snps Symlink the files into the analysis directory
In the APCL_analysis directory, create a new analysis folder mkdir 21-03seq
In all of the XXseq/sample directories, symlink the files to the 21-03seq directory
ln -s APCL* ../../APCL_analysis/21-03seq/
Wait for seq 19-21 to be ready
LS0tCnRpdGxlOiAiUHJvY2Vzc2luZyBzZXF1ZW5jZXMgZmFsbCAyMDE4IgpvdXRwdXQ6IGh0bWxfbm90ZWJvb2sKcGFyYW1zOgogIHNlcTogIFNFUTA3CiAgCi0tLQpDaGFuZ2VkIHRoZSBuYW1lcyBvZiB0aGUgc2VxdWVuY2VzIHRvIGFsbG93IGZvciB0aGUgc2VxdWVuY2luZyBpZHMgdGhhdCBTYXJhaCBoYXMgdXNlZC4KCmBgYHtyIHNldHVwfQpsaWJyYXJ5KHRpZHl2ZXJzZSkKc291cmNlKCIuLi9nZW5vbWljcy9zY3JpcHRzL2xhYl9oZWxwZXJzLlIiKQpsYWIgPC0gcmVhZF9kYigiTGFib3JhdG9yeSIpCmBgYApTZWUgdGhlIGp1cHl0ZXIgbm90ZWJvb2sgc2VxX3Byb2Nlc3NpbmcgZm9yIHRoZSBhbXBoaXByaW9uIHNpZGUgb2YgdGhpcyBwcm90b2NvbC4gIFN0YXJ0IHRoZXJlLgoKIyMjQ3JlYXRlIGFuIGluZGV4IGZpbGUgZm9yIHRoZSBQb29scyB0aGF0IGlzIHRhYiBzZXBhcmF0ZWQgKipkb25lKioKCmBgYHtyfQojIHJlYWQgYmFpdHMgdGFibGUKYmFpdHMgPC0gbGFiICU+JQogIHRibCgiYmFpdHMiKSAlPiUKICBjb2xsZWN0KCkgJT4lCiAgc2VsZWN0KGJhaXRzX2lkLCBzZXEpCgojIHJlYWQgUENSIHRhYmxlCnBjciA8LSBsYWIgJT4lIAogIHRibCgicGNyIikgJT4lIAogIGNvbGxlY3QoKSAlPiUgCiAgZmlsdGVyKFNFUSAlaW4lIHBhcmFtcyRzZXEpICU+JSAKICBzZWxlY3QocGNyX2lkLCBiYWl0X2lkLCBpbmRleCkKCiMgam9pbiBwY3JfaWQgYW5kIGluZGV4IHRvIHNlcSBpZApwb29scyA8LSBsZWZ0X2pvaW4ocGNyLCBiYWl0cywgYnkgPSBjKCJiYWl0X2lkIiA9ICJiYWl0c19pZCIpKSAlPiUgCiAgc2VsZWN0KHNlcSwgcGNyX2lkLCBpbmRleCkKCiMgcHVsbCBpbiB0aGUgYmFyY29kZXMgZm9yIGlsbHVtaW5hIGluZGV4ZXMKaW5kZXggPC0gbGFiICU+JSAKICB0YmwoImlsbHVtaW5hIikgJT4lIAogIGNvbGxlY3QoKQoKIyBqb2luIHRoZSBiYXJjb2RlcyB0byB0aGUgc2VxIGFuZCBwY3IgaWRzCnBvb2xzIDwtIGxlZnRfam9pbihwb29scywgaW5kZXgsIGJ5ID0gYygiaW5kZXgiID0gImluZGV4X251bSIpKSAlPiUgCiAgc2VsZWN0KHNlcSwgcGNyX2lkLCBpbmRleF9jb2RlKQoKIyBjcmVhdGUgYSBsaXN0IG9mIHRoZSBtdWx0aXBsZSBzZXFzCnNlcXMgPC0gc2VsZWN0KHBvb2xzLCBzZXEpICU+JSAKICBkaXN0aW5jdCgpCmZvcihpIGluIHNlcShzZXFzJHNlcSkpewogIHggPC0gcG9vbHMgJT4lIAogIGZpbHRlcihzZXEgPT0gc2VxcyRzZXFbaV0pICU+JSAKICBzZWxlY3QocGNyX2lkLCBpbmRleF9jb2RlKQoKIyB3cml0ZSB0aGUgZmlsZXMgZm9yIGFtcGhpcHJpb24KIyByZWFkcjo6d3JpdGVfdHN2KHgsIHBhdGggPSBwYXN0ZTAoImluZGV4LSIsIHNlcXMkc2VxW2ldLCAiLnRzdiIpLCBjb2xfbmFtZXMgPSBGKQp9CmBgYAoKIyNNb3ZlIHRoZSBmaWxlcyBmcm9tIHRoZSBsYXB0b3AgdG8gYW1waGlwcmlvbiB1c2luZyB0aGUgZmV0Y2ggcHJvZ3JhbSwgcGxhY2UgdGhlIGluZGV4IGZpbGVzIGluIHRoZSBsb2cgZm9sZGVyIGZvciBlYWNoIHNlcSAqKiBkb25lKiogIApTZWUgTWljaGVsbGUgaWYgeW91IG5lZWQgYSBzZXJpYWwgbnVtYmVyIGZvciBGZXRjaAoKCiMjI0NyZWF0ZSBhIG5hbWVzIGZpbGUgZm9yIGVhY2ggcG9vbCAqKmRvbmUqKgpUaGUgbmFtZXMgaGF2ZSB0byBiZSBzcGVjaWVzLCB1bmRlcnNjb3JlIGFuZCB0aGVuIHRoZSBzYW1wbGUgaWRlbnRpZmllciwgc28gQVBDTF9MNTQzMiBmb3IgbGlnYXRpb25faWQgTDU0MzIKYGBge3J9CiMgcmVhZCB0aGUgbGlnYXRpb25zCmxpZ3MgPC0gbGFiICU+JSAKICB0YmwoImxpZ2F0aW9uIikgJT4lIAogIGZpbHRlcihwb29sICVpbiUgcG9vbHMkcGNyX2lkKSAlPiUgCiAgc2VsZWN0KGxpZ2F0aW9uX2lkLCBiYXJjb2RlX251bSwgcG9vbCkgJT4lIAogIGNvbGxlY3QoKQoKIyByZWFkIHRoZSBiYXJjb2RlcwpiYXJjb2RlIDwtIGxhYiAlPiUgCiAgdGJsKCJiYXJjb2RlcyIpICU+JSAKICBjb2xsZWN0KCkKCiMgam9pbiB0aGUgbGlnYXRpb25faWQgYW5kIHBvb2wgaWRzIHRvIHRoZSBiYXJjb2RlcwpsaWdzIDwtIGxlZnRfam9pbihsaWdzLCBiYXJjb2RlLCBieSA9ICJiYXJjb2RlX251bSIpCgojIGpvaW4gdGhlIGxpZ3MgdG8gdGhlIHNlcV9pZHMKbGlncyA8LSBsZWZ0X2pvaW4obGlncywgcG9vbHMsIGJ5ID0gYygicG9vbCIgPSAicGNyX2lkIikpICU+JSAKICAjIGFkanVzdCB0aGUgbGlnYXRpb24gbmFtZSBmb3IgZERvY2VudAogIG11dGF0ZShuYW1lID0gcGFzdGUwKCJBUENMXyIsIGxpZ2F0aW9uX2lkKSkgJT4lIAogIHNlbGVjdChzZXEsIHBvb2wsIG5hbWUsIGJhcmNvZGUpICU+JSAKIyByZWR1Y2UgdGhlIHBvb2wgdG8gb25seSBhIG51bWJlcgogICAgbXV0YXRlKHBvb2wgPSBzdWJzdHIocG9vbCwgMiw1KSkKCiMgZm9yIHNlcTA0LCB0aGUgc2VxIGNvbHVtbiBpcyBlbXB0eS4gIFJlcGxhY2UgaXQuCmxpZ3MgPC0gbGlncyAlPiUgCiAgbXV0YXRlKHNlcSA9IHBhcmFtcyRzZXEpCgojIHdyaXRlIGZpbGVzIGZvciBhbXBoaXByaW9uCiMgY3JlYXRlIGEgbGlzdCBvZiB0aGUgbXVsdGlwbGUgc2VxcwpuYW1lcyA8LSBzZWxlY3QobGlncywgcG9vbCkgJT4lIAogIGRpc3RpbmN0KCkKCiMgbG9vcCB0aHJvdWdoIGFsbCBvZiB0aGUgcG9vbHMKZm9yKGkgaW4gc2VxKG5hbWVzJHBvb2wpKXsKICB4IDwtIGxpZ3MgJT4lIAogIGZpbHRlcihwb29sID09IG5hbWVzJHBvb2xbaV0pICU+JSAKICBzZWxlY3QobmFtZSwgYmFyY29kZSkKCiMgd3JpdGUgdGhlIGZpbGVzIGZvciBhbXBoaXByaW9uCnJlYWRyOjp3cml0ZV90c3YoeCwgcGF0aCA9IHBhc3RlMCgibmFtZXNfIiwgbmFtZXMkcG9vbFtpXSwgIi50c3YiKSwgY29sX25hbWVzID0gRikKfQoKYGBgCiMjI01vdmUgdGhlIGZpbGVzIGZyb20gdGhlIGxhcHRvcCB0byBhbXBoaXByaW9uIHVzaW5nIHRoZSBmZXRjaCBwcm9ncmFtLCBwbGFjZSB0aGUgbmFtZXMgZmlsZXMgaW4gdGhlIGxvZyBmb2xkZXIgZm9yIGVhY2ggc2VxICoqZG9uZSoqCgoKCgoKIyMjQ29uY2F0ZW5hdGUgdGhlIHJlc3VsdHMgaW4gdGhlIGJjc3BsaXQgZm9sZGVyIC0gdGFrZXMgYWJvdXQgYSBtaW51dGUgKipkb25lKioKCmBgYHtyfQpsaWJyYXJ5KHJlYWRyKQpmb3IgKGogaW4gc2VxcyRzZXEpewogIHkgPC0gcG9vbHMgJT4lIAogICAgZmlsdGVyKHNlcSA9PSBqKQogICMgd3JpdGVfZmlsZSgiIyEvYmluL2Jhc2giLCBwYXRoID0gcGFzdGUwKGosIl9jYXRfYWxsLnNoIiksIGFwcGVuZCA9IEYpCiAgIyBhZGQgYSBsaW5lIHJldHVybgogICMgd3JpdGVfZmlsZSgiXG4iLCBwYXRoID0gcGFzdGUwKGosIl9jYXRfYWxsLnNoIiksIGFwcGVuZCA9IFQpCiAgZm9yIChpIGluIHNlcSh5JHBjcl9pZCkpewogICAgeiA8LSBmb3JtYXRDKGksIHdpZHRoID0gMiwgZmxhZyA9IDApCiAgICAjIGdlbmVyYXRlIHRoZSBjb2RlCiAgICB4IDwtIHBhc3RlKCJjYXQiLCBwYXN0ZTAoIi4vbGFuZTEvIiwgeSRwY3JfaWRbaV0sICItcmVhZC0xLmZhc3RxLmd6IiksIHBhc3RlMCgiLi9sYW5lMi8iLCB5JHBjcl9pZFtpXSwgIi1yZWFkLTEuZmFzdHEuZ3oiKSwgIj4iLCAgcGFzdGUwKCIuLi8iLCB6LCAiUG9vbC8iLCB5JHBjcl9pZFtpXSwgIi5mYXN0cS5neiIpLCBzZXAgPSAiICIpCiAgICAjIGFkZCB0aGUgY29kZSB0byB0aGUgZmlsZQogICAgIyB3cml0ZV9maWxlKHgsIHBhdGggPSBwYXN0ZTAoaiwiX2NhdF9hbGwuc2giKSwgYXBwZW5kID0gVCkgIAogICAgIyBhZGQgYSBsaW5lIHJldHVybgogICAgIyB3cml0ZV9maWxlKCJcbiIsIHBhdGggPSBwYXN0ZTAoaiwiX2NhdF9hbGwuc2giKSwgYXBwZW5kID0gVCkKICB9ICAKfQoKYGBgCgoKIyMjIGNyZWF0ZSBzY3JpcHRzIGZvciBhbGwgNDggcG9vbHMgdG8gcHJvY2VzcyByYWR0YWdzICoqZG9uZSoqCgpgYGB7cn0KI2Fzc2lnbiB0aGUgY29ycmVjdCBwb29sIGRpcmVjdG9yeSB0byBlYWNoIHBvb2wKCnBvb2xzIDwtIHBvb2xzICU+JSAKICBhcnJhbmdlKHNlcSwgcGNyX2lkKSAlPiUgCiAgbXV0YXRlKHBvb2xfZGlyID0gcmVwKDE6MTIsIDQpLCAKICAgICAgICAgcG9vbF9kaXIgPSBmb3JtYXRDKHBvb2xfZGlyLCB3aWR0aCA9IDIsIGZsYWcgPSAwKSwgCiAgICAgICAgIHBvb2xfZGlyID0gcGFzdGUwKHBvb2xfZGlyLCAiUG9vbCIpKQoKIyMgd3JpdGUgZmlsZXMKbGlicmFyeShyZWFkcikKIyBjcmVhdGUgZmlyc3QgbGluZSBvZiBuZXcgZmlsZQpmb3IgKGkgaW4gc2VxKHBvb2xzJHNlcSkpewogICMgd3JpdGVfZmlsZSgiIyEvYmluL2Jhc2giLCBwYXRoID0gcGFzdGUwKHBvb2xzJHBjcl9pZFtpXSwgIl9wcm9jZXNzLnNoIiksIGFwcGVuZCA9IEYpCiAgIyBhZGQgYSBsaW5lIHJldHVybgogICMgd3JpdGVfZmlsZSgiXG4iLCBwYXRoID0gcGFzdGUwKHBvb2xzJHBjcl9pZFtpXSwgIl9wcm9jZXNzLnNoIiksIGFwcGVuZCA9IFQpCiAgIyBnZW5lcmF0ZSB0aGUgY29kZQogIHggPC0gcGFzdGUoInByb2Nlc3NfcmFkdGFncyAtYiAuL2xvZ3MvYmFyY29kZXMgLWMgLXEgLS1yZW56XzEgcHN0SSAtLXJlbnpfMiBtbHVDSSAtaSBnemZhc3RxIC0tYWRhcHRlcl8xIEFDQUNUQ1RUVENDQ1RBQ0FDR0FDR0NUQ1RUQ0NHQVRDVCAtZiIsIHBhc3RlMCgiLi8iLCBwb29scyRwb29sX2RpcltpXSwgIi8iLCBwb29scyRwY3JfaWRbaV0sICIuZmFzdHEuZ3oiKSwgICItbyIsIHBhc3RlMCgiLi8iLCBwb29scyRwb29sX2RpcltpXSwgIi8iKSwgc2VwID0gIiAiKQogICMgYWRkIHRoZSBjb2RlIHRvIHRoZSBmaWxlCiAgIyB3cml0ZV9maWxlKHgsIHBhdGggPSBwYXN0ZTAocG9vbHMkcGNyX2lkW2ldLCAiX3Byb2Nlc3Muc2giKSwgYXBwZW5kID0gVCkgIAp9CgpgYGAKIyMjTW92ZSB0aGUgc2NyaXB0cyB0byBhbXBoaXByaW9uIGludG8gdGhlIHNlcSBzY3JpcHRzIGZvbGRlciB3aXRoIGZldGNoIApXaXRoIGZldGNoIHlvdSBjYW4gYWxzbyBoaWdobGlnaHQgYWxsIG9mIHRoZSBzY3JpcHRzIGFuZCBjbGljayBnZXQgaW5mbywgZXhlY3V0ZSBpbnN0ZWFkIG9mIGhhdmluZyB0byB0eXBlIGluIHRoZSBjaG1vZCB1K3ggY29tbWFuZCBmb3IgZWFjaCBmaWxlLgoKI3Byb2Nlc3NfcmFkdGFncyAtIHVzZSBlc2MtUiBvciBjdHJsLVwgdG8gImZpbmQgYW5kIHJlcGxhY2UiIGluIG5hbm8gLSB0YWtlcyBhYm91dCAyLjUgaG91cnMgZm9yIDQgcG9vbHMgYW5kIDE5MiBzYW1wbGVzCgpGcm9tIHRoZSBzZXEgZm9sZGVyICgxOFNFUSBmb3IgZXhhbXBsZSksIHJ1biB0aGUgcHJvY2VzcyBzY3JpcHRzLCBjYW4gcnVuIGFsbCA0IGluIHNlcGFyYXRlIHdpbmRvd3MgKG9yIGFsbCA0OD8pCgojIHdyaXRlIGEgc2NyaXB0IHRvIGdlbmVyYXRlIGNvbW1hbmQgbGluZSB0byBydW4gcHJvY2VzcyByYWR0YWdzIHNjcmlwdHMKYGBge3J9CmZvciAoaiBpbiBzZXFzJHNlcSl7CiAgeSA8LSBwb29scyAlPiUgCiAgICBmaWx0ZXIoc2VxID09IGopCiAgd3JpdGVfZmlsZSgiIyEvYmluL2Jhc2giLCBwYXRoID0gcGFzdGUwKGosIl9wcm9jZXNzX2FsbC5zaCIpLCBhcHBlbmQgPSBGKQogICMgYWRkIGEgbGluZSByZXR1cm4KICB3cml0ZV9maWxlKCJcbiIsIHBhdGggPSBwYXN0ZTAoaiwiX3Byb2Nlc3NfYWxsLnNoIiksIGFwcGVuZCA9IFQpCiAgZm9yIChpIGluIHkkcGNyX2lkKXsKICAgICMgZ2VuZXJhdGUgdGhlIGNvZGUKICAgIHggPC0gcGFzdGUoIm5vaHVwIiwgcGFzdGUwKCIuL3NjcmlwdHMvIiwgaSwgIl9wcm9jZXNzLnNoICYiKSwgc2VwID0gIiAiKQogICAgIyBhZGQgdGhlIGNvZGUgdG8gdGhlIGZpbGUKICAgIHdyaXRlX2ZpbGUoeCwgcGF0aCA9IHBhc3RlMChqLCJfcHJvY2Vzc19hbGwuc2giKSwgYXBwZW5kID0gVCkgIAogICAgIyBhZGQgYSBsaW5lIHJldHVybgogICAgd3JpdGVfZmlsZSgiXG4iLCBwYXRoID0gcGFzdGUwKGosIl9wcm9jZXNzX2FsbC5zaCIpLCBhcHBlbmQgPSBUKQogIH0gIAp9CgpgYGAKCgpzdGFydGVkIHRoZSBsYXN0IG9uZSBhdCA5OjQ1cG0gb24gTW9uZGF5IDkvMjQvMjAxOCAtIHNob3VsZCB0YWtlIGFib3V0IDUgaG91cnMKZG9uZSBiZWZvcmUgNmFtIG9uIFR1ZXNkYXkKCgpXcml0ZSB0ZXh0IGZvciBtb3ZpbmcgZmlsZXMKYGBge3J9CmZvciAoaiBpbiBzZXFzJHNlcSl7CiAgeSA8LSBwb29scyAlPiUgCiAgICBmaWx0ZXIoc2VxID09IGopCiAgd3JpdGVfZmlsZSgiIyEvYmluL2Jhc2giLCBwYXRoID0gcGFzdGUwKGosIl9tb3ZlX3JhZGxvZ3Muc2giKSwgYXBwZW5kID0gRikKICAjIGFkZCBhIGxpbmUgcmV0dXJuCiAgd3JpdGVfZmlsZSgiXG4iLCBwYXRoID0gcGFzdGUwKGosIl9tb3ZlX3JhZGxvZ3Muc2giKSwgYXBwZW5kID0gVCkKICBmb3IgKGkgaW4gc2VxKHkkcGNyX2lkKSl7CiAgICB6IDwtIGZvcm1hdEMoaSwgd2lkdGggPSAyLCBmbGFnID0gMCkKICAgICMgZ2VuZXJhdGUgdGhlIGNvZGUKICAgIHggPC0gcGFzdGUoIm12IiwgcGFzdGUwKHosICJQb29sL3Byb2Nlc3NfcmFkdGFncy5sb2ciKSwgcGFzdGUwKCIuL2xvZ3MvcHJvY2VzcyIsIHkkcGNyX2lkW2ldLCAiLmxvZyIpLCBzZXAgPSAiICIpCiAgICAjIGFkZCB0aGUgY29kZSB0byB0aGUgZmlsZQogICAgd3JpdGVfZmlsZSh4LCBwYXRoID0gcGFzdGUwKGosIl9tb3ZlX3JhZGxvZ3Muc2giKSwgYXBwZW5kID0gVCkgIAogICAgIyBhZGQgYSBsaW5lIHJldHVybgogICAgd3JpdGVfZmlsZSgiXG4iLCBwYXRoID0gcGFzdGUwKGosIl9tb3ZlX3JhZGxvZ3Muc2giKSwgYXBwZW5kID0gVCkKICB9ICAKfQoKYGBgCgoKCldoaWNoIHNhbXBsZXMgZmFpbGVkPwpgYGB7cn0KdXBsaWdzIDwtIGxhYiAlPiUKICB0YmwoImxpZ2F0aW9uIikgJT4lIAogIGZpbHRlcihwb29sICVpbiUgY3VycmVudCRwY3JfaWQpICU+JSAKICBtdXRhdGUodG90YWxfcmVhZHMgPSBhcy5udW1lcmljKHRvdGFsX3JlYWRzKSwgCiAgICBsYWNrX3JhZF90YWcgPSBhcy5udW1lcmljKGxhY2tfcmFkX3RhZyksIAogICAgbG93X3F1YWxpdHkgPSBhcy5udW1lcmljKGxvd19xdWFsaXR5KSwgCiAgICByZXRhaW5lZCA9IGFzLm51bWVyaWMocmV0YWluZWQpKSAlPiUgCiAgY29sbGVjdCgpCgoKaGlzdCh1cGxpZ3MkdG90YWxfcmVhZHMsIGJyZWFrcyA9IDEwMCkKCiMgem9vbSBpbiB0byB0aGUgbG93ZXIgZW5kIAp6b29tIDwtIHVwbGlncyAlPiUgCiAgZmlsdGVyKHRvdGFsX3JlYWRzIDwgMjAwMDAwKQpoaXN0KHpvb20kdG90YWxfcmVhZHMsIGJyZWFrcyA9IDEwKQoKbG93IDwtIHVwbGlncyAlPiUgCiAgbXV0YXRlKHRvdGFsX3JlYWRzID0gYXMubnVtZXJpYyh0b3RhbF9yZWFkcykpICU+JSAKICBmaWx0ZXIodG90YWxfcmVhZHMgPCAxMDAwMDApICU+JSAKICBzZWxlY3QobGlnYXRpb25faWQpCgoKCnN1bW1hcnkodXBsaWdzKQoKbWF4aW11bSA8LSB1cGxpZ3MgJT4lIAogIGZpbHRlcihyZXRhaW5lZCA+IDEwMDAwMDApCgp0b3RhbCA8LSB1cGxpZ3MgJT4lIAogIHN1bW1hcmlzZShzdW0gPSBzdW0odG90YWxfcmVhZHMpKQoKYGBgCgoKCiMjIFJlbmFtZSB0aGUgcHJvY2VzcyByYWR0YWdzIG91dHB1dCB0byBzYW1wbGUgbmFtZXMKd3JpdGUgcmVuYW1pbmcgdGV4dApgYGB7cn0KbGlicmFyeShyZWFkcikKZm9yIChqIGluIHNlcXMkc2VxKXsKICB5IDwtIHBvb2xzICU+JSAKICAgIGZpbHRlcihzZXEgPT0gaikgJT4lIAogICAgYXJyYW5nZShwY3JfaWQpCiAgd3JpdGVfZmlsZSgiIyEvYmluL2Jhc2giLCBwYXRoID0gcGFzdGUwKGosIl9hbGxfcmVuYW1lLnNoIiksIGFwcGVuZCA9IEYpCiAgIyBhZGQgYSBsaW5lIHJldHVybgogIHdyaXRlX2ZpbGUoIlxuIiwgcGF0aCA9IHBhc3RlMChqLCJfYWxsX3JlbmFtZS5zaCIpLCBhcHBlbmQgPSBUKQogIGZvciAoaSBpbiBzZXEoeSRwY3JfaWQpKXsKICAgIHogPC0gZm9ybWF0QyhpLCB3aWR0aCA9IDIsIGZsYWcgPSAwKQogICAgYyA8LSBzdWJzdHIoeSRwY3JfaWRbaV0sIDIsIDUpCiAgICAjIGdlbmVyYXRlIHRoZSBjb2RlCiAgICBhIDwtIHBhc3RlKCJjZCIsIHBhc3RlMCh6LCAiUG9vbC8iKSwgc2VwID0gIiAiKQogICAgIyBhZGQgdGhlIGNvZGUgdG8gdGhlIGZpbGUKICAgIHdyaXRlX2ZpbGUoYSwgcGF0aCA9IHBhc3RlMChqLCJfYWxsX3JlbmFtZS5zaCIpLCBhcHBlbmQgPSBUKSAgCiAgICAjIGFkZCBhIGxpbmUgcmV0dXJuCiAgICB3cml0ZV9maWxlKCJcbiIsIHBhdGggPSBwYXN0ZTAoaiwiX2FsbF9yZW5hbWUuc2giKSwgYXBwZW5kID0gVCkKICAgIHggPC0gcGFzdGUoInNoIHJlbmFtZS5mb3IuZERvY2VudF9zZV9neiIsICBwYXN0ZTAoIi4uL2xvZ3MvbmFtZXNfIiwgYywgIi50c3YiKSwgc2VwID0gIiAiKQogICAgIyBhZGQgdGhlIGNvZGUgdG8gdGhlIGZpbGUKICAgIHdyaXRlX2ZpbGUoeCwgcGF0aCA9IHBhc3RlMChqLCJfYWxsX3JlbmFtZS5zaCIpLCBhcHBlbmQgPSBUKSAgCiAgICAjIGFkZCBhIGxpbmUgcmV0dXJuCiAgICB3cml0ZV9maWxlKCJcbiIsIHBhdGggPSBwYXN0ZTAoaiwiX2FsbF9yZW5hbWUuc2giKSwgYXBwZW5kID0gVCkKICAgIHdyaXRlX2ZpbGUoIm12IEFQQ0wqIC4uL3NhbXBsZXMvIiwgcGF0aCA9IHBhc3RlMChqLCJfYWxsX3JlbmFtZS5zaCIpLCBhcHBlbmQgPSBUKQogICAgd3JpdGVfZmlsZSgiXG4iLCBwYXRoID0gcGFzdGUwKGosIl9hbGxfcmVuYW1lLnNoIiksIGFwcGVuZCA9IFQpCiAgd3JpdGVfZmlsZSgiY2QgLi4iLCBwYXRoID0gcGFzdGUwKGosIl9hbGxfcmVuYW1lLnNoIiksIGFwcGVuZCA9IFQpICAKICB3cml0ZV9maWxlKCJcbiIsIHBhdGggPSBwYXN0ZTAoaiwiX2FsbF9yZW5hbWUuc2giKSwgYXBwZW5kID0gVCkKICAgIAogIH0gIAp9CgpgYGAKCgoKCgojIyBUcmltIGFuZCBtYXAgdGhlIHJlYWRzCm1pY2hlbGxlcyAyMDE4LTA5LTI1IDA3OjI1OjEwIHNhbXBsZXMgJCBkRG9jZW50CmREb2NlbnQgMi41LjUgCgpDb250YWN0IGpwdXJpdHpAdXJpLmVkdSB3aXRoIGFueSBwcm9ibGVtcyAKCiAKQ2hlY2tpbmcgZm9yIHJlcXVpcmVkIHNvZnR3YXJlCi9sb2NhbC9ob21lL21pY2hlbGxlcy8xNF9wcm9ncmFtcy9kRG9jZW50L2REb2NlbnQ6IGxpbmUgODQ6IFs6IDEuMC1yODI6IGludGVnZXIgZXhwcmVzc2lvbiBleHBlY3RlZAoKQWxsIHJlcXVpcmVkIHNvZnR3YXJlIGlzIGluc3RhbGxlZCEKCioqZERvY2VudCBydW4gc3RhcnRlZCBUdWUgU2VwIDI1IDA3OjI1OjE1IEVEVCAyMDE4ICoqCgo1NzYgaW5kaXZpZHVhbHMgYXJlIGRldGVjdGVkLiBJcyB0aGlzIGNvcnJlY3Q/IEVudGVyIHllcyBvciBubyBhbmQgcHJlc3MgW0VOVEVSXQp5ZXMKUHJvY2VlZGluZyB3aXRoIDU3NiBpbmRpdmlkdWFscwpkRG9jZW50IGRldGVjdHMgNDAgcHJvY2Vzc29ycyBhdmFpbGFibGUgb24gdGhpcyBzeXN0ZW0uClBsZWFzZSBlbnRlciB0aGUgbWF4aW11bSBudW1iZXIgb2YgcHJvY2Vzc29ycyB0byB1c2UgZm9yIHRoaXMgYW5hbHlzaXMuCjMwCmREb2NlbnQgZGV0ZWN0cyAyNTIgZ2lnYWJ5dGVzIG9mIG1heGltdW0gbWVtb3J5IGF2YWlsYWJsZSBvbiB0aGlzIHN5c3RlbS4KUGxlYXNlIGVudGVyIHRoZSBtYXhpbXVtIG1lbW9yeSB0byB1c2UgZm9yIHRoaXMgYW5hbHlzaXMgaW4gZ2lnYWJ5dGVzCkZvciBleGFtcGxlLCB0byBsaW1pdCBkRG9jZW50IHRvIHRlbiBnaWdhYnl0ZXMsIGVudGVyIDEwClRoaXMgb3B0aW9uIGRvZXMgbm90IHdvcmsgd2l0aCBhbGwgZGlzdHJpYnV0aW9ucyBvZiBMaW51eC4gIElmIHJ1bnMgYXJlIGhhbmdpbmcgYXQgdmFyaWFudCBjYWxsaW5nLCBlbnRlciAwClRoZW4gcHJlc3MgW0VOVEVSXQoxNTAKCkRvIHlvdSB3YW50IHRvIHF1YWxpdHkgdHJpbSB5b3VyIHJlYWRzPwpUeXBlIHllcyBvciBubyBhbmQgcHJlc3MgW0VOVEVSXT8KeWVzCgpEbyB5b3Ugd2FudCB0byBwZXJmb3JtIGFuIGFzc2VtYmx5PwpUeXBlIHllcyBvciBubyBhbmQgcHJlc3MgW0VOVEVSXS4Kbm8KClJlZmVyZW5jZSBjb250aWdzIG5lZWQgdG8gYmUgaW4gYSBmaWxlIG5hbWVkIHJlZmVyZW5jZS5mYXN0YQoKRG8geW91IHdhbnQgdG8gbWFwIHJlYWRzPyAgVHlwZSB5ZXMgb3Igbm8gYW5kIHByZXNzIFtFTlRFUl0KeWVzCkJXQSB3aWxsIGJlIHVzZWQgdG8gbWFwIHJlYWRzLiAgWW91IG1heSBuZWVkIHRvIGFkanVzdCAtQSAtQiBhbmQgLU8gcGFyYW1ldGVycyBmb3IgeW91ciB0YXhhLgpXb3VsZCB5b3UgbGlrZSB0byBlbnRlciBhIG5ldyBwYXJhbWV0ZXJzIG5vdz8gVHlwZSB5ZXMgb3Igbm8gYW5kIHByZXNzIFtFTlRFUl0KeWVzClBsZWFzZSBlbnRlciBuZXcgdmFsdWUgZm9yIEEgKG1hdGNoIHNjb3JlKS4gIEl0IHNob3VsZCBiZSBhbiBpbnRlZ2VyLiAgRGVmYXVsdCBpcyAxLgoxClBsZWFzZSBlbnRlciBuZXcgdmFsdWUgZm9yIEIgKG1pc21hdGNoIHNjb3JlKS4gIEl0IHNob3VsZCBiZSBhbiBpbnRlZ2VyLiAgRGVmYXVsdCBpcyA0Lgo0ClBsZWFzZSBlbnRlciBuZXcgdmFsdWUgZm9yIE8gKGdhcCBwZW5hbHR5KS4gIEl0IHNob3VsZCBiZSBhbiBpbnRlZ2VyLiAgRGVmYXVsdCBpcyA2Lgo2CkRvIHlvdSB3YW50IHRvIHVzZSBGcmVlQmF5ZXMgdG8gY2FsbCBTTlBzPyAgUGxlYXNlIHR5cGUgeWVzIG9yIG5vIGFuZCBwcmVzcyBbRU5URVJdCm5vCgpQbGVhc2UgZW50ZXIgeW91ciBlbWFpbCBhZGRyZXNzLiAgZERvY2VudCB3aWxsIGVtYWlsIHlvdSB3aGVuIGl0IGlzIGZpbmlzaGVkIHJ1bm5pbmcuCkRvbid0IHdvcnJ5OyBkRG9jZW50IGhhcyBubyBmaW5hbmNpYWwgbmVlZCB0byBzZWxsIHlvdXIgZW1haWwgYWRkcmVzcyB0byBzcGFtbWVycy4Kc3R1YXJ0NjIwQGdtYWlsLmNvbSAgICAgICAgCgoKQXQgdGhpcyBwb2ludCwgYWxsIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gaGFzIGJlZW4gZW50ZXJlZCBhbmQgZERvY2VudCBtYXkgdGFrZSBzZXZlcmFsIGhvdXJzIHRvIHJ1bi4KSXQgaXMgcmVjb21tZW5kZWQgdGhhdCB5b3UgbW92ZSB0aGlzIHNjcmlwdCB0byBhIGJhY2tncm91bmQgb3BlcmF0aW9uIGFuZCBkaXNhYmxlIHRlcm1pbmFsIGlucHV0IGFuZCBvdXRwdXQuCkFsbCBkYXRhIGFuZCBsb2dmaWxlcyB3aWxsIHN0aWxsIGJlIHJlY29yZGVkLgpUbyBkbyB0aGlzOgpQcmVzcyBjb250cm9sIGFuZCBaIHNpbXVsdGFuZW91c2x5ClR5cGUgJ2JnJyB3aXRob3V0IHRoZSBxdW90ZXMgYW5kIHByZXNzIGVudGVyClR5cGUgJ2Rpc293biAtaCcgYWdhaW4gd2l0aG91dCB0aGUgcXVvdGVzIGFuZCBwcmVzcyBlbnRlcgoKTm93IHNpdCBiYWNrLCByZWxheCwgYW5kIHdhaXQgZm9yIHlvdXIgYW5hbHlzaXMgdG8gZmluaXNoCgpGaW5pc2hlZCBhdCA5OjQ3YW0gc2FtZSBkYXkKCk9uY2UgdHJpbW1pbmcgaXMgZG9uZSwgcHJlcGFyZSB0byBjYWxsIHNucHMKU3ltbGluayB0aGUgZmlsZXMgaW50byB0aGUgYW5hbHlzaXMgZGlyZWN0b3J5CgpJbiB0aGUgQVBDTF9hbmFseXNpcyBkaXJlY3RvcnksIGNyZWF0ZSBhIG5ldyBhbmFseXNpcyBmb2xkZXIKbWtkaXIgMjEtMDNzZXEKCkluIGFsbCBvZiB0aGUgWFhzZXEvc2FtcGxlIGRpcmVjdG9yaWVzLCBzeW1saW5rIHRoZSBmaWxlcyB0byB0aGUgMjEtMDNzZXEgZGlyZWN0b3J5CgpsbiAtcyBBUENMKiAuLi8uLi9BUENMX2FuYWx5c2lzLzIxLTAzc2VxLwoKCgpXYWl0IGZvciBzZXEgMTktMjEgdG8gYmUgcmVhZHkgCgoK