forked from ioBroker/ioBroker
-
Notifications
You must be signed in to change notification settings - Fork 0
/
installer.sh
468 lines (398 loc) · 13.7 KB
/
installer.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
#!/bin/bash
# Increase this version number whenever you update the installer
INSTALLER_VERSION="2019-11-29" # format YYYY-MM-DD
# Test if this script is being run as root or not
if [[ $EUID -eq 0 ]];
then IS_ROOT=true; SUDOX=""
else IS_ROOT=false; SUDOX="sudo "; fi
ROOT_GROUP="root"
USER_GROUP="$USER"
# TODO: Change this URL when merging into stable!
LIB_NAME="installer_library.sh"
LIB_URL="https://raw.githubusercontent.com/ioBroker/ioBroker/master/$LIB_NAME"
# get and load the LIB
curl -sL $LIB_URL > ~/$LIB_NAME
if test -f ~/$LIB_NAME; then source ~/$LIB_NAME; else echo "Installer/Fixer: library not found"; exit -2; fi
# test one function of the library
RET=$(get_lib_version)
if [ $? -ne 0 ]; then echo "Installer/Fixer: library $LIB_NAME could not be loaded!"; exit -2; fi
if [ "$RET" == "" ]; then echo "Installer/Fixer: library $LIB_NAME does not work."; exit -2; fi
echo "Library version=$RET"
# Test which platform this script is being run on
get_platform_params
set_some_common_params
if [ "$IS_ROOT" = "true" ]; then
print_bold "Welcome to the ioBroker installer!" "Installer version: $INSTALLER_VERSION"
else
print_bold "Welcome to the ioBroker installer!" "Installer version: $INSTALLER_VERSION" "" "You might need to enter your password a couple of times."
fi
# Which npm package should be installed (default "iobroker")
INSTALL_TARGET=${INSTALL_TARGET-"iobroker"}
export AUTOMATED_INSTALLER="true"
NUM_STEPS=4
# ########################################################
print_step "Installing prerequisites" 1 "$NUM_STEPS"
# update repos
$SUDOX $INSTALL_CMD update
# Install Node.js if it is not installed
if [[ $(which "node" 2>/dev/null) != *"/node" ]]; then
install_nodejs
fi
# Check if npm is installed
if [[ $(which "npm" 2>/dev/null) != *"/npm" ]]; then
# If not, try to install it
install_package npm
if [[ $(which "npm" 2>/dev/null) != *"/npm" ]]; then
echo "${red}Cannot continue because \"npm\" is not installed and could not be installed automatically!${normal}"
exit 1
fi
fi
# Select an npm mirror, by default use npmjs.org
REGISTRY_URL="https://registry.npmjs.org"
case "$MIRROR" in
[Tt]aobao)
REGISTRY_URL="https://registry.npm.taobao.org"
;;
esac
if [ $(npm config get registry) != "$REGISTRY_URL" ]; then
echo "Changing npm registry to $REGISTRY_URL"
npm config set registry $REGISTRY_URL
fi
# Determine the platform we operate on and select the installation routine/packages accordingly
install_necessary_packages
# ########################################################
print_step "Creating ioBroker user and directory" 2 "$NUM_STEPS"
# Ensure the user "iobroker" exists and is in the correct groups
if [ "$HOST_PLATFORM" = "linux" ]; then
create_user_linux $IOB_USER
elif [ "$HOST_PLATFORM" = "freebsd" ]; then
create_user_freebsd $IOB_USER
fi
# Ensure the installation directory exists and take control of it
$SUDOX mkdir -p $IOB_DIR
if [ "$IS_ROOT" != true ]; then
# During the installation we need to give the current user access to the install dir
# On Linux, we'll fix this at the end. On OSX this is okay
if [ "$HOST_PLATFORM" = "osx" ]; then
sudo chown -R $USER $IOB_DIR
else
sudo chown -R $USER:$USER_GROUP $IOB_DIR
fi
fi
cd $IOB_DIR
echo "Directory $IOB_DIR created"
# Log some information about the installer
touch $INSTALLER_INFO_FILE
chmod 777 $INSTALLER_INFO_FILE
echo "Installer version: $INSTALLER_VERSION" >> $INSTALLER_INFO_FILE
echo "Installation date $(date +%F)" >> $INSTALLER_INFO_FILE
echo "Platform: $HOST_PLATFORM" >> $INSTALLER_INFO_FILE
# ########################################################
print_step "Installing ioBroker" 3 "$NUM_STEPS"
# Disable any warnings related to "npm audit fix"
disable_npm_audit
if [ "$HOST_PLATFORM" = "freebsd" ]; then
# Make sure we use the correct python binary
set_npm_python
fi
# download the installer files and run them
# If this script is run as root, we need the --unsafe-perm option
if [ "$IS_ROOT" = true ]; then
echo "Installed as root" >> $INSTALLER_INFO_FILE
npm i $INSTALL_TARGET --loglevel error --unsafe-perm > /dev/null
else
echo "Installed as non-root user $USER" >> $INSTALLER_INFO_FILE
npm i $INSTALL_TARGET --loglevel error > /dev/null
fi
npm i --production --loglevel error --unsafe-perm > /dev/null
# ########################################################
print_step "Finalizing installation" 4 "$NUM_STEPS"
# Test which init system is used:
INITSYSTEM="unknown"
if [[ "$HOST_PLATFORM" = "freebsd" && -d "/usr/local/etc/rc.d" ]]; then
INITSYSTEM="rc.d"
elif [[ `systemctl` =~ -\.mount ]] &> /dev/null; then
INITSYSTEM="systemd"
elif [[ -f /etc/init.d/cron && ! -h /etc/init.d/cron ]]; then
INITSYSTEM="init.d"
elif [[ "$HOST_PLATFORM" = "osx" ]]; then
INITSYSTEM="launchctl"
PLIST_FILE_LABEL="org.ioBroker.LaunchAtLogin"
SERVICE_FILENAME="/Users/${IOB_USER}/Library/LaunchAgents/${PLIST_FILE_LABEL}.plist"
fi
if [[ $IOB_FORCE_INITD && ${IOB_FORCE_INITD-x} ]]; then
INITSYSTEM="init.d"
fi
echo "init system: $INITSYSTEM" >> $INSTALLER_INFO_FILE
# #############################
# Create "iob" and "iobroker" executables
# If possible, try to always execute the iobroker CLI as the correct user
IOB_NODE_CMDLINE="node"
if [ "$IOB_USER" != "$USER" ]; then
IOB_NODE_CMDLINE="sudo -H -u $IOB_USER node"
fi
if [ "$INITSYSTEM" = "systemd" ]; then
# systemd needs a special executable that reroutes iobroker start/stop to systemctl
# Make sure to only use systemd when there is exactly 1 argument
IOB_EXECUTABLE=$(cat <<- EOF
#!$BASH_CMDLINE
if (( \$# == 1 )) && ([ "\$1" = "start" ] || [ "\$1" = "stop" ] || [ "\$1" = "restart" ]); then
sudo systemctl \$1 iobroker
elif [ "\$1" = "fix" ]; then
curl -sL $FIXER_URL | bash -
else
$IOB_NODE_CMDLINE $CONTROLLER_DIR/iobroker.js \$@
fi
EOF
)
elif [ "$INITSYSTEM" = "launchctl" ]; then
# launchctl needs unload service to stop iobroker
IOB_EXECUTABLE=$(cat <<- EOF
#!$BASH_CMDLINE
if (( \$# == 1 )) && ([ "\$1" = "start" ]); then
launchctl load -w $SERVICE_FILENAME
elif (( \$# == 1 )) && ([ "\$1" = "stop" ]); then
launchctl unload -w $SERVICE_FILENAME
$IOB_NODE_CMDLINE $CONTROLLER_DIR/iobroker.js stop
elif [ "\$1" = "fix" ]; then
curl -sL $FIXER_URL | bash -
else
$IOB_NODE_CMDLINE $CONTROLLER_DIR/iobroker.js \$@
fi
EOF
)
else
IOB_EXECUTABLE=$(cat <<- EOF
#!$BASH_CMDLINE
if [ "\$1" = "fix" ]; then
curl -sL $FIXER_URL | bash -
else
$IOB_NODE_CMDLINE $CONTROLLER_DIR/iobroker.js \$@
fi
EOF
)
fi
if [ "$HOST_PLATFORM" = "linux" ]; then
IOB_BIN_PATH=/usr/bin
elif [ "$HOST_PLATFORM" = "freebsd" ] || [ "$HOST_PLATFORM" = "osx" ]; then
IOB_BIN_PATH=/usr/local/bin
fi
# Symlink the global binaries iob and iobroker
$SUDOX ln -sfn $IOB_DIR/iobroker $IOB_BIN_PATH/iobroker
$SUDOX ln -sfn $IOB_DIR/iobroker $IOB_BIN_PATH/iob
# Symlink the local binary iob
$SUDOX ln -sfn $IOB_DIR/iobroker $IOB_DIR/iob
# Create executables in the ioBroker directory
# TODO: check if this must be fixed like in in the FIXER for #216
write_to_file "$IOB_EXECUTABLE" $IOB_DIR/iobroker
make_executable "$IOB_DIR/iobroker"
# TODO: check if this is necessary, like in the FIXER
## and give them the correct ownership
#change_owner $IOB_USER "$IOB_DIR/iobroker"
#change_owner $IOB_USER "$IOB_DIR/iob"
# #############################
# Enable autostart
# From https://unix.stackexchange.com/questions/18209/detect-init-system-using-the-shell/326213
# if [[ `/sbin/init --version` =~ upstart ]]; then echo using upstart;
# elif [[ `systemctl` =~ -\.mount ]]; then echo using systemd;
# elif [[ -f /etc/init.d/cron && ! -h /etc/init.d/cron ]]; then echo using sysv-init;
# else echo cannot tell; fi
# Enable autostart
if [[ "$INITSYSTEM" = "init.d" ]]; then
echo "Enabling autostart..."
# Write a script into init.d that automatically detects the correct node executable and runs ioBroker
INITD_FILE=$(cat <<- EOF
#!$BASH_CMDLINE
### BEGIN INIT INFO
# Provides: iobroker.sh
# Required-Start: \$network \$local_fs \$remote_fs
# Required-Stop: \$network \$local_fs \$remote_fs
# Should-Start: redis-server
# Should-Stop: redis-server
# Default-Start: 2 3 4 5
# Default-Stop: 0 1 6
# Short-Description: starts ioBroker
# Description: starts ioBroker
### END INIT INFO
PIDF=$CONTROLLER_DIR/lib/iobroker.pid
NODECMD=\$(which node)
RETVAL=0
start() {
echo -n "Starting ioBroker"
su - $IOB_USER -s "$BASH_CMDLINE" -c "\$NODECMD $CONTROLLER_DIR/iobroker.js start"
RETVAL=\$?
}
stop() {
echo -n "Stopping ioBroker"
su - $IOB_USER -s "$BASH_CMDLINE" -c "\$NODECMD $CONTROLLER_DIR/iobroker.js stop"
RETVAL=\$?
}
if [ "\$1" = "start" ]; then
start
elif [ "\$1" = "stop" ]; then
stop
elif [ "\$1" = "restart" ]; then
stop
start
else
echo "Usage: iobroker \{start\|stop\|restart\}"
exit 1
fi
exit \$RETVAL
EOF
)
# Create the startup file, give it the correct permissions and start ioBroker
SERVICE_FILENAME="/etc/init.d/iobroker.sh"
write_to_file "$INITD_FILE" $SERVICE_FILENAME
set_root_permissions $SERVICE_FILENAME
$SUDOX bash $SERVICE_FILENAME
echo "Autostart enabled!"
# Remember what we did
if [[ $IOB_FORCE_INITD && ${IOB_FORCE_INITD-x} ]]; then
echo "Autostart: init.d (forced)" >> "$INSTALLER_INFO_FILE"
else
echo "Autostart: init.d" >> "$INSTALLER_INFO_FILE"
fi
elif [ "$INITSYSTEM" = "systemd" ]; then
echo "Enabling autostart..."
# Write an systemd service that automatically detects the correct node executable and runs ioBroker
SYSTEMD_FILE=$(cat <<- EOF
[Unit]
Description=ioBroker Server
Documentation=http://iobroker.net
After=network.target redis.service influxdb.service mysql-server.service mariadb-server.service
Wants=redis.service
[Service]
Type=simple
User=$IOB_USER
Environment="NODE=\$(which node)"
ExecStart=$BASH_CMDLINE -c '\${NODE} $CONTROLLER_DIR/controller.js'
Restart=on-failure
[Install]
WantedBy=multi-user.target
EOF
)
# Create the startup file and give it the correct permissions
SERVICE_FILENAME="/lib/systemd/system/iobroker.service"
write_to_file "$SYSTEMD_FILE" $SERVICE_FILENAME
if [ "$IS_ROOT" != true ]; then
sudo chown root:$ROOT_GROUP $SERVICE_FILENAME
fi
$SUDOX chmod 644 $SERVICE_FILENAME
$SUDOX systemctl daemon-reload
$SUDOX systemctl enable iobroker
$SUDOX systemctl start iobroker
echo "Autostart enabled!"
echo "Autostart: systemd" >> "$INSTALLER_INFO_FILE"
elif [ "$INITSYSTEM" = "rc.d" ]; then
echo "Enabling autostart..."
PIDFILE="$CONTROLLER_DIR/lib/iobroker.pid"
# Write an rc.d service that automatically detects the correct node executable and runs ioBroker
RCD_FILE=$(cat <<- EOF
#!$BASH_CMDLINE
#
# PROVIDE: iobroker
# REQUIRE: DAEMON
# KEYWORD: shutdown
. /etc/rc.subr
name="iobroker"
rcvar="iobroker_enable"
load_rc_config \$name
iobroker_enable=\${iobroker_enable-"NO"}
iobroker_pidfile=\${iobroker_pidfile-"$PIDFILE"}
iobroker_start()
{
iobroker start
}
iobroker_stop()
{
iobroker stop
}
iobroker_status()
{
iobroker status
}
PATH="\${PATH}:/usr/local/bin"
pidfile="\${iobroker_pidfile}"
start_cmd=iobroker_start
stop_cmd=iobroker_stop
status_cmd=iobroker_status
run_rc_command "\$1"
EOF
)
# Create the startup file, give it the correct permissions and start ioBroker
SERVICE_FILENAME="/usr/local/etc/rc.d/iobroker"
write_to_file "$RCD_FILE" $SERVICE_FILENAME
set_root_permissions $SERVICE_FILENAME
# Make sure that $IOB_USER may access the pidfile
$SUDOX touch "$PIDFILE"
$SUDOX chown $IOB_USER:$IOB_USER $PIDFILE
# Enable startup and start the service
sysrc iobroker_enable=YES
service iobroker start
echo "Autostart enabled!"
echo "Autostart: rc.d" >> "$INSTALLER_INFO_FILE"
elif [ "$INITSYSTEM" = "launchctl" ]; then
echo "Enabling autostart..."
NODECMD=$(which node)
# osx use launchd.plist init system.
PLIST_FILE=$(cat <<- EOF
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>Label</key>
<string>${PLIST_FILE_LABEL}</string>
<key>ProgramArguments</key>
<array>
<string>${NODECMD}</string>
<string>${CONTROLLER_DIR}/iobroker.js</string>
<string>start</string>
</array>
<key>KeepAlive</key>
<true/>
<key>RunAtLoad</key>
<true/>
<key>EnvironmentVariables</key>
<dict>
<key>PATH</key>
<string>/usr/local/bin:/usr/local/sbin:/usr/bin:/bin:/usr/sbin:/sbin</string>
</dict>
</dict>
</plist>
EOF
)
# Create the startup file, give it the correct permissions and start ioBroker
echo "$PLIST_FILE" > $SERVICE_FILENAME
# Enable startup and start the service
launchctl list ${PLIST_FILE_LABEL} &> /dev/null
if [ $? -eq 0 ]; then
echo "Reloading service ${PLIST_FILE_LABEL}"
launchctl unload -w $SERVICE_FILENAME
fi
launchctl load -w $SERVICE_FILENAME
echo "Autostart enabled!"
echo "Autostart: launchctl" >> "$INSTALLER_INFO_FILE"
else
echo "${yellow}Unsupported init system, cannot enable autostart!${normal}"
echo "Autostart: false" >> "$INSTALLER_INFO_FILE"
fi
# Test again which platform this script is being run on
# This is necessary because FreeBSD does crazy stuff
get_platform_params
# Make sure that the app dir belongs to the correct user
# Don't do it on OSX, because we'll install as the current user anyways
if [ "$HOST_PLATFORM" != "osx" ]; then
fix_dir_permissions
fi
# Force npm to run as iobroker when inside IOB_DIR
if [[ "$IS_ROOT" != true && "$USER" != "$IOB_USER" ]]; then
change_npm_command_user
fi
change_npm_command_root
unset AUTOMATED_INSTALLER
# Detect IP address
IP=$(detect_ip_address)
print_bold "${green}ioBroker was installed successfully${normal}" "Open http://$IP:8081 in a browser and start configuring!"
print_msg "${yellow}You need to re-login before doing anything else on the console!${normal}"
exit 0