I want to make the current file location as the working directory.
Using Rstudio (Works!):
# Author : Bhishan Poudel
# Program :
Try using parent.frame(3)
in you function:
setwd_thisdir <- function () {
this.dir <- dirname(parent.frame(3)$ofile)
setwd(this.dir)
}
See ?parent.frame
or http://adv-r.had.co.nz/Environments.html#calling-environments.
You may also look at the chdir
option of the source
function (?source
).
UPDATE: I realised that this answer didn't help at all, and I will post another one that does the trick.
Insofar the code you want to run doesn't need any additional arguments, a solution as sketched below, using eval(expr, envir)
might do the trick.
Consider the following example using print(environment())
, which should return environment: R_GlobalEnv
when used on the command line. The function test_1
will print information about the internal environment that is created when the function is called, whereas the function test_2
will return the desired result.
test_1 <- function(){
print(environment())
}
test_2 <- function(){
.expr <- quote({
print(environment())
})
.envir <- sys.frame(which = -1)
eval(expr = .expr,
envir = .envir)
}
The sys.frame(which = -1)
ensures that the expression is evaluated in the environment where the function is called. If you are certain that you always want to use the global environment, then it's better to use .GlobalEnv
. It's also important to quote the expression you want to use, otherwise it might not work as desired.
A nice feature of this solution is that you don't need to tweak the code you want to put into the function, just quote it.
Finally: It's possible to extend this approach such that your function can take arguments that then will be given to the code you want to evaluate in another environment. This will however require a bit of non-trivial tweaking upon the expression you want to evaluate; you might need to use the bquote
+ .()
construction - and you might in addition also need to use call
and do.call
.
Simply, use rstudio API, extract its directory, and set it as a working directory as shown below:
setwd(dirname(rstudioapi::getSourceEditorContext()$path))
Verify if you set the directory correctly by the following command:
getwd()
I write another answer because you changed your question. There are two useful facts:
ofile
is a variable in the environment of the source
function, so you can use it only when you run some script with the source
function.So, to comment on your observations:
source
function), but not if you press Run (which just runs the commands in the R console).ofile
without a call to source
.this_dir <- function(directory) setwd( file.path(getwd(), directory) )
is needless as it is just the definition of a function called this_dir
.setwd_thisdir
is needless because it just prints the body of setwd_thisdir
to the console.In summary setwd(dirname(parent.frame(2)$ofile))
is a useful trick when you source a script with the source
function, but don't have access to the source
function options: e.g. when you press Source in R-Studio. When possible use intead the source
function with chdir=TRUE
. If you run the script form the terminal just set the terminal to the script folder.
The first answer I gave missed the point completely, since I hadn't looked closely upon what you wanted to achieve. The solution presented here should however do the trick.
First note that source
has an argument chdir
that in the help-file is described with: logical; if TRUE
and file
is a pathname, the R working directory is temporarily changed to the directory containing file
for evaluating.
To manually specify that argument every time you want to source a file
would be a pain, so let's add something to .Rprofile that changes the
default value for chdir
from FALSE
to TRUE
.
The formals
-function can be used to modify a default
value, but when used upon a function that belongs to some other
environment, the result will be that a local copy of the function will be created instead. That's not good enough.
It's probably several ways to resolve this, but the following
little hack of source
did the trick for me when I inserted it into .Rprofile.
.temporary_copy_source <- base::source
formals(.temporary_copy_source)$chdir <- TRUE
utils::assignInNamespace(
x = "source",
value = .temporary_copy_source,
ns = environment(source))
rm(.temporary_copy_source)
A word of warning: The method presented here can in principle allow users to modify the default values of any argument in any function, but that would be an exceptionally bad idea to do. Keep in mind that your scripts might later on be shared with someone that doesn't have the same .Rprofile that you have. Never write code that requires such modifications of the namespaces!