192

I would like to set a job to run daily in the root crontab. But I would like it to execute it from a particular directory so it can find all the files it needs, since the application has a bunch of relative paths.

Anyway, can I tell crontab to run from a particular directory?

Jav_Rock
  • 21,011
  • 18
  • 115
  • 164
user333746
  • 2,085
  • 3
  • 15
  • 11

2 Answers2

368

All jobs are executed by a shell, so start that shell snippet by a command to change the directory.

cd /path/to/directory && ./bin/myapp

Concerning the use of && instead of ;: normally it doesn't make a difference, but if the cd command fails (e.g. because the directory doesn't exist) with && the application isn't executed, whereas with ; it's executed (but not in the intended directory).

lfurini
  • 3,420
  • 4
  • 27
  • 40
Gilles 'SO- stop being evil'
  • 92,660
  • 35
  • 189
  • 229
  • Rather than add a mostly duplicate answer, let me just add that you can choose which shell (if you need bash rather than sh, for example) by setting `SHELL` in your crontab. – Edd Steel Jan 17 '12 at 18:42
  • So this command would be to execute the app (stored in bin) in another directory, wouldn't it? But if you already have the app installed in the right directory, the command I put in my answer would be ok, right? Sorry, I just wan't to understand the question. – Jav_Rock Jan 17 '12 at 18:43
  • 15
    +1. every other related question's solution is to re-write the script with absolute paths. this is exactly what I needed – Conrad.Dean Jun 22 '12 at 15:53
  • 3
    Out of curiosity, could you use a semi-colon `;` instead of the double ampersands `&&` or would that not work? And why? – Joshua Pinter Apr 19 '16 at 22:51
  • 35
    @JoshPinter Normally it doesn't make a difference. But if the `cd` command fails (e.g. because the directory doesn't exist), with `&&`, the application isn't executed. With `;`, it's executed (but not in the intended directory). – Gilles 'SO- stop being evil' Apr 19 '16 at 23:05
  • 1
    I will do this to all the cron jobs of my apps , you saved me from changing my source files, thank you very much – Accountant م Mar 27 '19 at 19:39
2

Reading man 5 crontab should tell you that there's a HOME variable set which can be redefined in the file. It becomes your working directory. You can set PATH for the command(s) too. Of course this affects all the cron schedule lines.

E.G.

Several environment variables are set up automatically by the cron(8) daemon. SHELL is set to /bin/sh, and LOGNAME and HOME are set from the /etc/passwd line of the crontab´s owner. HOME and SHELL can be overridden by settings in the crontab; LOGNAME can not.

(Note: the LOGNAME variable is sometimes called USER on BSD systems and is also automatically set).

Depending on your cron of course, but mine also has MAILTO, MAILFROM CONTENT_TYPE, CRON_TZ, RANDOM_DELAY, and MLS_LEVEL.

So for your hypothetical app I'd recommend a file name /etc/cron.d/hypothetical containing:

# Runs hypothetical app @ 00:01Z in its local path for reading its config or something.
SHELL=/bin/sh
HOME=/where/the/app/is
PATH=/where/the/app/is:/bin:/usr/local/bin:/usr/bin:/bin:/usr/sbin:/sbin
CRON_TZ=UTC
1 0 * * * theappuser hypothetical --with arguments 

For example with docker-compose relying on the cwd docker-compose.yml:

SHELL=/bin/sh
HOME=/path/to/composed-app
5 5 * * * root docker-compose restart -t 10 service-name
dlamblin
  • 40,676
  • 19
  • 92
  • 127