Skip to contents
library(GWASBrewer)
library(dplyr)
#> 
#> Attaching package: 'dplyr'
#> The following objects are masked from 'package:stats':
#> 
#>     filter, lag
#> The following objects are masked from 'package:base':
#> 
#>     intersect, setdiff, setequal, union
library(ggplot2)
set.seed(1)

Introduction

In some applications, we may need to generate multiple data sets for the same set of effect sizes. This would mimic performing multiple GWAS on the same trait. There are two functions, resample_sumstats and resample_inddata that facilitate this. This vignette will demonstrate these functions as well as discuss important considerations when the simulation involves sampling GWAS for multiple ancestries for the same trait.

To get started, we will generate a small sim_mv object. For a review of that function, see the “Simulating Data” vignette. For this vignette, we will keep things very simple by having only 12 variants with very large effect sizes. Of course, this is not realistic, but it will let us see exactly what is going on. In our example, we will have two traits. The first trait has an effect of 0.2 on the second trait. Recalling that sim_mv always sets the variance of each trait to 1, this means, that trait 1 explains 4% (\(0.2^2 = 0.04\)) of the variance of trait 2. We will also use an LD pattern. Here we just specify the LD pattern for five variants. This will be repeated 2.2 times to cover 12 variants. We will specify N so that there is no sample overlap.

set.seed(1000)
my_ld1 <- matrix(c( 1.00,  0.60,  0.40, -0.1, -0.07,
                 0.60,  1.00,  0.60, -0.1, -0.07,
                 0.40,  0.60,  1.00, -0.1, -0.07,
                 -0.10, -0.10, -0.10,  1.0,  0.90,
                 -0.07, -0.07, -0.07,  0.9,  1.00), nrow = 5, byrow = T)
af1 <- c(0.35, 0.3, 0.4, 0.72, 0.75)

G <- matrix(c(0, 0.2, 0, 0), nrow = 2, byrow =T) # matrix of causal effects
orig_dat <- sim_mv(N = c(10000, 20000),
                 J = 12,
                 h2 = c(0.2, 0.3),
                 pi = 0.4,
                 G = G,
                 R_LD = list(my_ld1),
                 af = af1, 
                 est_s = TRUE)
#> SNP effects provided for 12 SNPs and 2 traits.

Resampling Summary Statistics or Individual Level Data from the Same Population

In the simplest scenario, we want to simulate performing a new GWAS or collecting new individual level data for the original set of traits in the original population, meaning that the LD pattern and allele frequencies are the same.

Resampling Summary Statistics from the Same Population

To resample summary statistics, we can use resample_sumstats. Here, we will regenerate data assuming that samples for the new GWAS are totally overlapping.

N1 <- matrix(50000, nrow = 2, ncol = 2)
new_dat1 <- resample_sumstats(dat = orig_dat,
                              N = N1, 
                              R_LD = list(my_ld1), 
                              af = af1, 
                              est_s = TRUE)
#> I will assume that the environmental variance is the same in the old and new population.
#> I will assume that environmental correlation is the same in the old and new population. Note that this could result in different overall trait correlations.
#> SNP effects provided for 12 SNPs and 2 traits.

The new_dat1 object is another object of class sim_mv. The new and old data have the same effect sizes so beta_joint, beta_marg, direct_SNP_effects_joint, and direct_SNP_effects_marg are all the same between the two objects. They also have the same causal effect structure so the trait effect matrices direct_trait_effects and total_trait_effects are the same. However, the summary statistic matrices beta_hat and s_estimate are different due to random sampling variation. The true standard errors, se_beta_hat are also different because the sample size is different. Finally, since there is sample overlap in the samples for new_dat1 but not in orig_dat, the R matrices which give the correlation of effect sizes across traits are different.

all.equal(orig_dat$beta_joint, new_dat1$beta_joint)
#> [1] TRUE
all.equal(orig_dat$beta_marg, new_dat1$beta_marg)
#> [1] TRUE

head(orig_dat$beta_hat)
#>               [,1]         [,2]
#> [1,] -0.1080910136 -0.009750299
#> [2,] -0.1571366506  0.013286133
#> [3,] -0.1125931870  0.004225466
#> [4,]  0.2322018052  0.020554680
#> [5,]  0.2105222770  0.019324228
#> [6,]  0.0002105625 -0.299937884
head(new_dat1$beta_hat)
#>              [,1]         [,2]
#> [1,] -0.106418649  0.019449404
#> [2,] -0.169053933  0.029430270
#> [3,] -0.101005542  0.009112222
#> [4,]  0.249366373  0.050046188
#> [5,]  0.233752764  0.051418782
#> [6,] -0.002706624 -0.291194649

head(orig_dat$se_beta_hat)
#>            [,1]       [,2]
#> [1,] 0.01482499 0.01048285
#> [2,] 0.01543033 0.01091089
#> [3,] 0.01443376 0.01020621
#> [4,] 0.01574852 0.01113589
#> [5,] 0.01632993 0.01154701
#> [6,] 0.01482499 0.01048285
head(new_dat1$se_beta_hat)
#>             [,1]        [,2]
#> [1,] 0.006629935 0.006629935
#> [2,] 0.006900656 0.006900656
#> [3,] 0.006454972 0.006454972
#> [4,] 0.007042952 0.007042952
#> [5,] 0.007302967 0.007302967
#> [6,] 0.006629935 0.006629935

head(orig_dat$s_estimate)
#>            [,1]       [,2]
#> [1,] 0.01462257 0.01053189
#> [2,] 0.01514042 0.01091440
#> [3,] 0.01432211 0.01025373
#> [4,] 0.01563268 0.01119651
#> [5,] 0.01623509 0.01164215
#> [6,] 0.01471137 0.01025486
head(new_dat1$s_estimate)
#>             [,1]        [,2]
#> [1,] 0.006612584 0.006626085
#> [2,] 0.006862624 0.006899731
#> [3,] 0.006454305 0.006467005
#> [4,] 0.006970514 0.007052383
#> [5,] 0.007221306 0.007289617
#> [6,] 0.006621972 0.006489576

Resampling Individual Level Data from the Same Population

We can also generate individual level data. For this we use the resample_inddata function. This function can be used to produce three types of output:

  • Genotypes only
  • Genotypes and phenotypes
  • Phenotypes only

resample_inddata uses the hapsim package to generate individual level data. The sample size argument, N, in resample_inddata can accept three types of input: scalar, vector, or data frame. These formats are the same as those used by sim_mv, so if N is a scalar or vector, resample_inddata will assume there is no sample overlap. This means that if we use N = 10 with two traits, we will get out genotype information for 20 individuals, of which half have phenotype 1 information and half have phenotype 2 information. To specify sample overlap, we can use the data frame format. As a reminder, a sample size data frame should have columns named trait_1, … trait_[M] and N. The trait_[x] columns will be interpreted as logicals and the N column should give the number of samples in each combination of studies.

Generating genotypes only

If resample_inddata is used to generate genotypes only, we are not really “resampling” anything because there is no need to input trait data. In this case, we only need to supply the total number of individuals, the number of variants, and the LD pattern if desired. Here we will generate data for 15 individuals with an LD pattern matching the one used in orig_dat. The returned data includes the genotype matrix and a vector of population allele frequencies.

genos_only <- resample_inddata(N = 15, 
                               J = 12,
                               R_LD = list(my_ld1), 
                               af = af1)
#> Loading required package: hapsim
#> Loading required package: MASS
#> 
#> Attaching package: 'MASS'
#> The following object is masked from 'package:dplyr':
#> 
#>     select
#> Generating genotype matrix only.
names(genos_only)
#> [1] "X"  "af"
dim(genos_only$X)
#> [1] 15 12
length(genos_only$af)
#> [1] 12
genos_only$X
#>     [,1] [,2] [,3] [,4] [,5] [,6] [,7] [,8] [,9] [,10] [,11] [,12]
#> S1     0    0    0    2    2    1    1    1    2     2     1     0
#> S2     1    1    0    2    2    0    0    0    2     2     1     1
#> S3     0    0    0    2    2    1    0    2    2     2     1     1
#> S4     0    0    1    0    0    0    0    1    0     0     2     2
#> S5     2    1    1    2    2    1    0    0    0     0     0     0
#> S6     1    0    2    1    1    0    0    0    2     2     1     0
#> S7     0    0    2    2    2    1    0    0    1     1     2     2
#> S8     0    0    0    2    2    0    1    0    2     2     2     2
#> S9     1    0    0    2    2    0    0    0    2     2     2     2
#> S10    2    2    2    1    1    1    1    1    1     1     2     2
#> S11    1    0    1    2    2    2    2    2    1     1     1     1
#> S12    0    1    1    2    2    0    2    2    2     2     1     1
#> S13    0    0    1    1    1    2    2    2    1     1     1     1
#> S14    0    0    0    1    1    1    1    0    2     2     1     1
#> S15    0    1    0    2    2    1    1    2    2     2     2     2

Generating Genotypes and Phenotypes

To generate both genotypes and phenotypes, we need to include a sim_mv object that contains effect sizes. For our example, we will use the data frame sample size format to indicate that three individuals have both phenotypes measured and seven have only one or the other measured. Note that we don’t need to include the J argument because the sim_mv object contains all the necessary information. The vector of M phenotypes for each individual is simulated as \(Y_i = Y_{G,i} + Y_{E,i}\) where \(Y_{G,i}\) is the vector of genetic components computed from the genotypes and effect sizes stored in the original sim_mv object and \(Y_{E,i}\) is a vector of environmental components sampled from a normal distribution with variance given by the Sigma_E matrix in the original sim_mv object.

N <- data.frame("trait_1" = c(1, 1, 0), "trait_2" = c(0, 1, 1), "N" = c(3, 3, 4))
new_ind_dat1 <- resample_inddata(N = N, 
                                 dat = orig_dat, 
                                 R_LD = list(my_ld1), 
                                 af = af1, 
                                 calc_sumstats = FALSE)
#> Generating both genotypes and phenotypes.
#> SNP effects provided for 12 SNPs and 2 traits.
#> I will assume that the environmental variance is the same in the old and new population.
#> I will assume that environmental correlation is the same in the old and new population. Note that this could result in different overall trait correlations.

Genotype data are stored in new_ind_dat1$X and phenotype data are in new_ind_dat1$Y.

names(new_ind_dat1)
#> [1] "X"          "Y"          "af"         "Sigma_G"    "Sigma_E"   
#> [6] "pheno_sd"   "h2"         "trait_corr" "beta_joint"
new_ind_dat1$X
#>     [,1] [,2] [,3] [,4] [,5] [,6] [,7] [,8] [,9] [,10] [,11] [,12]
#> S1     0    0    1    1    1    0    0    0    1     1     1     2
#> S2     2    2    2    1    1    0    0    0    2     2     2     1
#> S3     0    0    1    1    1    1    1    1    0     0     0     0
#> S4     0    0    1    2    2    0    0    0    1     1     0     1
#> S5     0    0    0    2    2    1    0    0    2     2     1     1
#> S6     0    0    0    2    2    1    1    0    1     2     1     1
#> S7     1    1    1    1    1    2    1    1    2     2     2     1
#> S8     1    0    1    2    2    0    1    1    2     2     1     1
#> S9     1    1    2    1    1    2    2    2    0     0     2     2
#> S10    0    0    0    1    1    0    0    0    2     2     1     1
new_ind_dat1$Y
#>           y_1         y_2
#> 1  -0.9660107          NA
#> 2  -1.6707661          NA
#> 3  -0.4127266          NA
#> 4   0.3558235  1.57133493
#> 5   0.5363123  0.05390283
#> 6  -1.2896112  0.12940470
#> 7          NA  0.48527806
#> 8          NA -2.00566078
#> 9          NA -2.01050960
#> 10         NA  0.97844193

The returned object also includes some additional information including Sigma_G, Sigma_E, and beta_joint which have the same meaning as the objects with the same name in a sim_mv object. If we had set calc_sumstats to TRUE, the object would also include effect size and standard error estimates for the association between each variant and each trait.

Generating Phenotypes Only

Finally, we can use resample_inddata to calculate phenotypes for a previously generated set of genotypes from a sim_mv object. To do this, we supply the genotypes matrix to the genos argument. If this is supplied, no genotypes matrix will be returned. Even though the genotype matrix is supplied, we also need to supply the R_LD and af arguments for the population LD and allele frequencies. These are used to correctly calculate the genetic variance-covariance matrix. Note that it is important that the total number of individuals implied by the sample size argument match the number of rows in the genotypes matrix. The following call will generate an error because we supplied 15 genotypes but the sample size data frame only includes 10 individuals.

phenos_only <- resample_inddata(N = N, 
                                dat = orig_dat, 
                                genos = genos_only$X,
                                R_LD = list(my_ld1),
                                af = af1,
                                calc_sumstats = FALSE)
#> Generating phenotypes only.
#> Error in check_matrix(genos, "genos", ntotal, J): Expected genos to have 10 rows, found 15

We can fix the poblem by only including the first 10 rowsof the genotype matrix.

phenos_only <- resample_inddata(N = N, 
                                dat = orig_dat, 
                                genos = genos_only$X[1:10,],
                                R_LD = list(my_ld1),
                                af = af1,
                                calc_sumstats = FALSE)
#> Generating phenotypes only.
#> SNP effects provided for 12 SNPs and 2 traits.
#> I will assume that the environmental variance is the same in the old and new population.
#> I will assume that environmental correlation is the same in the old and new population. Note that this could result in different overall trait correlations.
names(phenos_only)
#> [1] "Y"          "af"         "Sigma_G"    "Sigma_E"    "pheno_sd"  
#> [6] "h2"         "trait_corr" "beta_joint"
phenos_only$Y
#>           y_1        y_2
#> 1  -0.6443696         NA
#> 2   0.7978988         NA
#> 3  -0.2142834         NA
#> 4  -0.2455013 -0.3309879
#> 5  -0.4448521 -0.9023457
#> 6  -0.1022008 -0.6312573
#> 7          NA -0.1918427
#> 8          NA  1.0820581
#> 9          NA -0.4664964
#> 10         NA -0.7820182

Resampling Data from a Different Population

Both resample_sumstats and resample_inddata can be used as above with different LD and allele frequencies than used to create the original object. However, there are some special considerations about trait variance and scaling that we will discuss in this section.

Understanding Effect Size Units

To understand resampled data that use different LD or allele frequencies than the original data, it is important to understand the units of effect sizes produced by sim_mv. sim_mv always assumes that phenotype variance is equal to 1, so the interpretation of the effects in beta_joint is the change in phenotype in units of phenotype SD per change in genotype in units of either allele or genotype SD. To see the genotype unit, we can check the geno_scale object. For example,

orig_dat$geno_scale
#> [1] "allele"

By default, sim_mv will use the per-allele scale if allele frequencies are available and otherwise use the per-SD scale. We can also check the the phenotypes are scaled to unit variance in pheno_sd which will always be a vector of 1’s if the object was produced by sim_mv.

orig_dat$pheno_sd
#> [1] 1 1

Changing LD and Allele Frequencies

The resampling functions in GWASBrewer assume that effect sizes in the new population are the same as in the old population on the genotype and phenotype scale given. If you are changing populations, it is much more sensible to assume that effect sizes are constant on the per-allele scale than on the per-genotype SD scale. This means that if you want to regenerate data from a different population, it is a good idea to start with input data that have geno_scale equal to “allele”.

One consequence of differing allele frequencies and LD structure is that the total genetic variance will be different in the new population than in the old. By default, the resampling functions will assume that the environmental variance is the same in the two populations. This means that the overall variance of the phenotype in the new population will probably not be 1. The resampling functions will not rescale the phenotype in the new population because this would mean that the phenotypes in the two populations had different units and were not comparable. However, you can rescale the output after the fact using if you would like the phenotypes to have unit variance.

We can see this in action by resampling summary statistics using different allele frequencies and LD pattern than used originally.

af2 <- rep(0.1, 5)
my_ld2 <- matrix(0.3, nrow = 5, ncol = 5)
diag(my_ld2) <- 1
new_dat2 <- resample_sumstats(dat = orig_dat, 
                              N = N, 
                              R_LD = list(my_ld2), 
                              af = af2)
#> Genetic variance in the new population differs from the genetic variance in the old population.
#> I will assume that the environmental variance is the same in the old and new population.
#> I will assume that environmental correlation is the same in the old and new population. Note that this could result in different overall trait correlations.
#> Note that the phenotype in the new population has a different variance from the phenotype in the old population.
#> I will keep the phenotype on the same scale as the original data, so effect sizes in the old and new object are comparable. If you would like to rescale the phenotype to have variance 1, use rescale_sumstats.
#> SNP effects provided for 12 SNPs and 2 traits.

Notice that the function produces a message to let us know that the phenotype has a different variance in the new population. We can check by looking at pheno_sd. The genetic covariance matrix, heritability, and overall trait correlation will also be different. However, the environmental covariance will be the same.

new_dat2$pheno_sd
#> [1] 0.9649663 0.9164662
orig_dat$pheno_sd
#> [1] 1 1
new_dat2$h2
#> [1] 0.04683348 0.11545049
orig_dat$h2
#> [1] 0.1124495 0.2570578
new_dat2$Sigma_G
#>             [,1]        [,2]
#> [1,] 0.043609461 0.008093447
#> [2,] 0.008093447 0.096968051
orig_dat$Sigma_G
#>            [,1]       [,2]
#> [1,] 0.11244947 0.01934268
#> [2,] 0.01934268 0.25705783
new_dat2$Sigma_E
#>           [,1]      [,2]
#> [1,] 0.8875505 0.1736201
#> [2,] 0.1736201 0.7429422
orig_dat$Sigma_E
#>           [,1]      [,2]
#> [1,] 0.8875505 0.1736201
#> [2,] 0.1736201 0.7429422

Note that the features that are kept constant are the environmental variance Sigma_E and the per-allele joint effects, beta_joint.

Changing Environmental Variance or Covariance

Although the default behavior is to keep Sigma_E constant, we can change this using the arguments new_env_var, new_h2, new_R_E and new_R_obs. These are options for both resample_sumstats and resample_inddata. The new_env_var and new_h2 arguments provide different ways of specifying the environmental variance so only one of these can be used at a time. new_h2 gives the heritability in the new population while new_env_var directly gives the environmental variance. The parameters new_R_E and new_R_obs are alternate ways of specifying environmental correlation. new_R_E directly specifies the enviromental correlation while new_R_obs gives the total trait correlation. Only one of these two parameters can be supplied.

new_R_E <- diag(2)
new_R_E[1,2] <- new_R_E[2,1] <- 0.4
new_dat3 <- resample_sumstats(dat = orig_dat, 
                              N = N, 
                              R_LD = list(my_ld2), 
                              af = af2,
                              new_env_var = c(0.9, 1.3), 
                              new_R_E = new_R_E)
#> Genetic variance in the new population differs from the genetic variance in the old population.
#> Note that the phenotype in the new population has a different variance from the phenotype in the old population.
#> I will keep the phenotype on the same scale as the original data, so effect sizes in the old and new object are comparable. If you would like to rescale the phenotype to have variance 1, use rescale_sumstats.
#> SNP effects provided for 12 SNPs and 2 traits.
new_dat3$Sigma_E
#>           [,1]      [,2]
#> [1,] 0.9000000 0.4326662
#> [2,] 0.4326662 1.3000000
cov2cor(new_dat3$Sigma_E)
#>      [,1] [,2]
#> [1,]  1.0  0.4
#> [2,]  0.4  1.0
new_dat3$h2
#> [1] 0.04621558 0.06941322

Note that specifying new_h2 will always result in exactly the heritability specified. This is in contrast with the h2 argument of sim_mv which gives the expected heritability.

new_dat4 <- resample_sumstats(dat = orig_dat, 
                              N = N, 
                              R_LD = list(my_ld2), 
                              af = af2,
                              new_h2 = c(0.15, 0.25), 
                              new_R_E = new_R_E)
#> Genetic variance in the new population differs from the genetic variance in the old population.
#> Note that the phenotype in the new population has a different variance from the phenotype in the old population.
#> I will keep the phenotype on the same scale as the original data, so effect sizes in the old and new object are comparable. If you would like to rescale the phenotype to have variance 1, use rescale_sumstats.
#> SNP effects provided for 12 SNPs and 2 traits.
new_dat4$h2
#> [1] 0.15 0.25
new_dat4$Sigma_E
#>           [,1]      [,2]
#> [1,] 0.2471203 0.1072480
#> [2,] 0.1072480 0.2909042

Rescaling Effect Size Units

If for any reason we want to change the units of effects in an object produced by sim_mv or resmaple_sumstats, we can use rescale_sumstats. For example, in the code below, we rescale the effects in orig_dat to be on the per-SD scale. Note that by doing this, we will delete the allele frequency information in the snp_info table. This is because allele frequency is irrelevant for per-SD scaled effects. To convert back to per-allele scale, we would need to supply the allele frequency.


orig_rescale1 <- rescale_sumstats(dat = orig_dat, 
                                  output_geno_scale = "sd")
orig_rescale1$beta_joint
#>               [,1]          [,2]
#>  [1,]  0.000000000  0.0000000000
#>  [2,] -0.097136125  0.0188220320
#>  [3,]  0.000000000  0.0000000000
#>  [4,]  0.146918495  0.0293836989
#>  [5,]  0.000000000  0.0000000000
#>  [6,]  0.000000000  0.0061492991
#>  [7,] -0.003777444 -0.0007554889
#>  [8,]  0.000000000 -0.5047783058
#>  [9,]  0.000000000  0.0000000000
#> [10,]  0.000000000  0.0000000000
#> [11,]  0.000000000  0.0000000000
#> [12,] -0.280286489 -0.0560572978
orig_rescale1$geno_scale
#> [1] "sd"
orig_dat$beta_joint
#>               [,1]         [,2]
#>  [1,]  0.000000000  0.000000000
#>  [2,] -0.149884295  0.029043026
#>  [3,]  0.000000000  0.000000000
#>  [4,]  0.231374881  0.046274976
#>  [5,]  0.000000000  0.000000000
#>  [6,]  0.000000000  0.009116327
#>  [7,] -0.005828723 -0.001165745
#>  [8,]  0.000000000 -0.728584727
#>  [9,]  0.000000000  0.000000000
#> [10,]  0.000000000  0.000000000
#> [11,]  0.000000000  0.000000000
#> [12,] -0.432491442 -0.086498288

## back to per-allele scale
orig_rescale2 <- rescale_sumstats(dat = orig_rescale1, 
                                  output_geno_scale = "allele", 
                                  af = orig_dat$snp_info$AF)
orig_rescale2$beta_joint
#>               [,1]         [,2]
#>  [1,]  0.000000000  0.000000000
#>  [2,] -0.149884295  0.029043026
#>  [3,]  0.000000000  0.000000000
#>  [4,]  0.231374881  0.046274976
#>  [5,]  0.000000000  0.000000000
#>  [6,]  0.000000000  0.009116327
#>  [7,] -0.005828723 -0.001165745
#>  [8,]  0.000000000 -0.728584727
#>  [9,]  0.000000000  0.000000000
#> [10,]  0.000000000  0.000000000
#> [11,]  0.000000000  0.000000000
#> [12,] -0.432491442 -0.086498288

We can also change the scale of the outcomes.

orig_rescale3 <- rescale_sumstats(dat = orig_dat, 
                                  output_geno_scale = "allele", 
                                  output_pheno_sd = c(1.4, 0.8))

There are a few effects of changing the phenotype scale. First the effect sizes are different. In this case the effect sizes for trait 1 have been multiplied by 1.4 and the effect sizes for trait 2 have been multiplied by 0.8. Second, the genetic and environmental covariance matrices have been scaled appropriately.

orig_rescale3$Sigma_G
#>            [,1]       [,2]
#> [1,] 0.22040097 0.02166381
#> [2,] 0.02166381 0.16451701
orig_rescale3$Sigma_E
#>           [,1]      [,2]
#> [1,] 1.7395990 0.1944545
#> [2,] 0.1944545 0.4754830
orig_rescale3$Sigma_G + orig_rescale3$Sigma_E
#>           [,1]      [,2]
#> [1,] 1.9600000 0.2161183
#> [2,] 0.2161183 0.6400000

Note that the heritability, orig_rescale3$h2 has not changed but is no longer equal to the diagonal of Sigma_G. The trait correlation in trait_corr is also the same. The final difference is in the total and direct trait effects matrix. In our case, trait 1 has been multiplied by 1.4 and trait 2 was multiplied by 0.8. On the original scale, a one unit increase in trait 1 caused a 0.2 unit increase in trait 2. So on the new scale, a 1 unit increase in trait 1 causes a \(0.2 \cdot (0.8/1.4) = 0.114\) unit increase in trait 2.

Note that the causal relationship between the traits implies the following relationship between effect sizes:

with(orig_dat, all.equal(direct_SNP_effects_joint[,2] + total_trait_effects[1,2]*direct_SNP_effects_joint[,1], beta_joint[,2]))
#> [1] TRUE

This is the basis of methods like Mendelian randomization. We can see that after scaling, this relationship is still true.

with(orig_rescale3, all.equal(direct_SNP_effects_joint[,2] + total_trait_effects[1,2]*direct_SNP_effects_joint[,1], beta_joint[,2]))
#> [1] TRUE