build 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420
  1. #!/bin/bash
  2. [% c("var/set_default_env") -%]
  3. distdir=/var/tmp/dist/[% project %]
  4. export TORBROWSER_VERSION='[% c("version") %]'
  5. mkdir -p $distdir
  6. scripts_dir=/var/tmp/build_scripts
  7. mkdir -p "$scripts_dir"
  8. OUTDIR='[% dest_dir _ "/" _ c("filename") %]'
  9. mkdir -p $OUTDIR
  10. # When we build with MULTI_LINGUAL=1, the browser will be packaged inside a
  11. # directory named tor-browser (instead of tor-browser_en-US). Therefore we
  12. # stage everything under tor-browser-stage to avoid a conflict.
  13. TB_STAGE_DIR=$distdir/tor-browser-stage
  14. GENERATEDPREFSPATH=$rootdir/Bundle-Data/PTConfigs/generated-prefs.js
  15. # Create initially empty prefs file where we can dump our conditionally included/genetered prefs
  16. touch "$GENERATEDPREFSPATH"
  17. [% IF c("var/osx") %]
  18. TBDIR="$TB_STAGE_DIR/Tor Browser.app"
  19. DOCSPATH=Contents/Resources/TorBrowser/Docs
  20. EXTSPATH=Contents/Resources/distribution/extensions
  21. TORBINPATH=Contents/MacOS/Tor
  22. TORCONFIGPATH=Contents/Resources/TorBrowser/Tor
  23. tar -C /var/tmp/dist -xf $rootdir/[% c('input_files_by_name/libdmg') %]
  24. export PATH=/var/tmp/dist/libdmg-hfsplus:$PATH
  25. [% ELSE %]
  26. TBDIR=$TB_STAGE_DIR/Browser
  27. DOCSPATH=TorBrowser/Docs
  28. EXTSPATH=TorBrowser/Data/Browser/profile.default/extensions
  29. TORCONFIGPATH=TorBrowser/Data/Tor
  30. mkdir -p "$TBDIR/TorBrowser/Data/Browser/Caches"
  31. [% END %]
  32. mkdir -p "$TBDIR/$EXTSPATH"
  33. # Extract the MAR tools.
  34. unzip -d $rootdir $rootdir/[% c('input_files_by_name/firefox') %]/mar-tools-*.zip
  35. MARTOOLS=$rootdir/mar-tools
  36. mv [% c('input_files_by_name/noscript') %] "$TBDIR/$EXTSPATH/{73a6fe31-595d-460b-a920-fcc0f8843232}.xpi"
  37. tar -C "$TBDIR" -xf [% c('input_files_by_name/obfs4') %]
  38. tar -C "$TBDIR" -xf [% c('input_files_by_name/snowflake') -%]
  39. tar -C "$TBDIR[% IF c("var/osx") %]/Contents/Resources[% END %]" -xf [% c('input_files_by_name/fonts') %]
  40. [% IF c("var/linux") %]
  41. cp RelativeLink/* ${TB_STAGE_DIR}/Browser/
  42. cp RelativeLink/start-tor-browser.desktop ${TB_STAGE_DIR}/
  43. cat > ${TB_STAGE_DIR}/Browser/start-tor-browser << 'RBM_TB_EOF'
  44. [% INCLUDE 'RelativeLink/start-tor-browser' -%]
  45. RBM_TB_EOF
  46. [% IF c("var/namecoin") %]
  47. pushd ${TB_STAGE_DIR}/Browser/
  48. patch -p1 < $rootdir/namecoin.patch
  49. popd
  50. [% END %]
  51. chmod +x ${TB_STAGE_DIR}/Browser/start-tor-browser
  52. # Make sure we get the desired scrollbar behavior with Gtk3, see bug 27546.
  53. GTK_SETTINGS_DIR=${TB_STAGE_DIR}/Browser/.config/gtk-3.0
  54. mkdir -p $GTK_SETTINGS_DIR
  55. cp $rootdir/gtk3-settings.ini $GTK_SETTINGS_DIR/settings.ini
  56. [% END %]
  57. tar -C ${TB_STAGE_DIR} -xf [% c('input_files_by_name/firefox') %]/tor-browser.tar.gz
  58. # Make unpacked HTTPS Everywhere available in resource://torbutton/content/extensions/https-everywhere/ as TB expects
  59. TMP_HTTPS_EVERYWHERE_PATH=$rootdir/tmp_https_everywhere/
  60. mkdir $TMP_HTTPS_EVERYWHERE_PATH
  61. pushd $TMP_HTTPS_EVERYWHERE_PATH
  62. mkdir -p chrome/torbutton/content/extensions/https-everywhere/
  63. unzip $rootdir/[% c('input_files_by_name/https-everywhere') %] -d chrome/torbutton/content/extensions/https-everywhere/
  64. find . -exec [% c("var/touch") %] {} \;
  65. find chrome/ | sort | zip -X -@ "$TBDIR[% IF c("var/osx") %]/Contents/Resources[% END %]/omni.ja"
  66. popd
  67. rm -rf $TMP_HTTPS_EVERYWHERE_PATH
  68. tar -C "$TBDIR[% IF ! c("var/osx") %]/TorBrowser[% END %]" -xf [% c('input_files_by_name/tor') %]/tor.tar.gz
  69. [% IF c("var/namecoin") %]
  70. # Extract Electrum-NMC
  71. tar -C "$TBDIR/TorBrowser" -xf [% c('input_files_by_name/electrum-nmc') %]
  72. # Extract ncprop279
  73. mkdir "$TBDIR/TorBrowser/ncprop279"
  74. tar -C "$TBDIR/TorBrowser/ncprop279" -xf [% c('input_files_by_name/ncprop279') %]
  75. # Extract StemNS
  76. tar -C "$TBDIR/TorBrowser" -xf [% c('input_files_by_name/stemns') %]
  77. [% END %]
  78. [% IF c("var/linux");
  79. SET bundledata_osname = 'linux';
  80. ELSIF c("var/osx");
  81. SET bundledata_osname = 'mac';
  82. ELSIF c("var/windows");
  83. SET bundledata_osname = 'windows';
  84. END; %]
  85. [% IF c("var/osx") %]
  86. # The Bundle-Data is designed for embedded data, so we need to modify
  87. # the structure when we want the data to be outside the app directory.
  88. # We also create an override.ini file to disable the profile migrator.
  89. SKELETON_TMP=$rootdir/Bundle-Data/mac-skeleton-tmp
  90. SKELETON_TMP_RESOURCES=$SKELETON_TMP/Contents/Resources
  91. mkdir -p $SKELETON_TMP_RESOURCES/browser
  92. echo "[XRE]" > $SKELETON_TMP_RESOURCES/browser/override.ini
  93. echo "EnableProfileMigrator=0" >> $SKELETON_TMP_RESOURCES/browser/override.ini
  94. mkdir -p $SKELETON_TMP_RESOURCES/TorBrowser/Tor
  95. cp -p Bundle-Data/mac/TorBrowser/Data/Tor/torrc-defaults $SKELETON_TMP_RESOURCES/TorBrowser/Tor/
  96. # Place a copy of the bookmarks.html file at the top. It will be moved into
  97. # browser/omni.ja later
  98. cp -p Bundle-Data/mac/TorBrowser/Data/Browser/profile.default/bookmarks.html $rootdir
  99. rm -Rf Bundle-Data/mac
  100. mv $SKELETON_TMP Bundle-Data/mac
  101. # Install a "tor" shim that sets the working directory. See #10030.
  102. mv "$TBDIR/$TORBINPATH/tor" "$TBDIR/$TORBINPATH/tor.real"
  103. cp Bundle-Data/mac-tor.sh "$TBDIR/$TORCONFIGPATH/tor"
  104. tar -C Bundle-Data/mac-applications.dmg -c . | tar -C $TB_STAGE_DIR -x
  105. [% END %]
  106. mkdir -p "$TBDIR/$DOCSPATH"
  107. cp -a Bundle-Data/Docs/* "$TBDIR/$DOCSPATH"
  108. tar -C Bundle-Data/[% bundledata_osname %] [% IF ! c("var/namecoin") %]--exclude=*Electrum-NMC* --exclude=*ncprop279*[% END %] -c . | tar -C "$TBDIR[% IF ! c("var/osx") %]/TorBrowser[% END %]" -x
  109. cat Bundle-Data/PTConfigs/[% bundledata_osname %]/torrc-defaults-appendix >> "$TBDIR/$TORCONFIGPATH/torrc-defaults"
  110. cat Bundle-Data/PTConfigs/bridge_prefs.js >> "$GENERATEDPREFSPATH"
  111. [% IF ! c("var/multi_lingual") %]
  112. echo 'pref("extensions.torlauncher.prompt_for_locale", false);' >> "$GENERATEDPREFSPATH"
  113. [% END %]
  114. [% IF c("var/linux") %]
  115. chmod 700 ${TB_STAGE_DIR}/Browser/TorBrowser/Data/Browser
  116. chmod 700 ${TB_STAGE_DIR}/Browser/TorBrowser/Data/Tor
  117. [% END %]
  118. [% IF c("var/multi_lingual") %]
  119. # If we are building a multi-lingual package, use "ALL" in the package name
  120. # and as the locale for update purposes. We do not include "ALL" in the name
  121. # of the directory that is inside the package (in other words, users will
  122. # not see tor-browser_ALL after they install Tor Browser).
  123. PKG_LOCALE="ALL"
  124. PKG_DIR="tor-browser"
  125. [% ELSE %]
  126. PKG_LOCALE="en-US"
  127. PKG_DIR="tor-browser_${PKG_LOCALE}"
  128. [% END %]
  129. pushd "$TBDIR[% IF c("var/osx") %]/Contents/Resources[% END %]/browser/"
  130. unzip omni.ja defaults/preferences/000-tor-browser.js || [ $? -lt 3 ]
  131. # Append our built extension-overrides.js to 000-tor-browser.js
  132. cat "$GENERATEDPREFSPATH" >> defaults/preferences/000-tor-browser.js
  133. cp defaults/preferences/000-tor-browser.js $rootdir
  134. [% IF c("var/osx") %]
  135. # Embed our default bookmarks within the en-US locale.
  136. mkdir -p chrome/en-US/locale/browser
  137. cp -p $rootdir/bookmarks.html chrome/en-US/locale/browser/
  138. [% c("var/touch") %] chrome/en-US/locale/browser/bookmarks.html
  139. chmod 600 chrome/en-US/locale/browser/bookmarks.html
  140. zip -Xm omni.ja chrome/en-US/locale/browser/bookmarks.html
  141. rm -rf chrome
  142. [% END %]
  143. # Set the locale of the bundle.
  144. echo "pref(\"intl.locale.requested\", \"en-US\");" >> defaults/preferences/000-tor-browser.js
  145. [% c("var/touch") %] defaults/preferences/000-tor-browser.js
  146. zip -Xm omni.ja defaults/preferences/000-tor-browser.js
  147. rm -rf defaults
  148. # create tbb_version.json file for #25020
  149. echo '{"version":"[% c("var/torbrowser_version") %]","architecture":"[% c("var/mar_osname") %]","channel":"[% c("var/channel") %]","locale":"en-US"}' > ../tbb_version.json
  150. popd
  151. [% IF c("var/osx") || c("var/multi_lingual") -%]
  152. # Prepare our language packs to embed our default bookmarks.
  153. # See bug 21879 for more details.
  154. [% SET locales = c("var/testbuild") ? [] : c("var/locales") -%]
  155. [% FOREACH lang = locales %]
  156. [% SET lang = tmpl(lang);
  157. SET xpi = '$rootdir/' _ c('input_files_by_name/firefox-langpacks') _ '/' _ lang _ '.xpi';
  158. %]
  159. [% IF c("var/osx") -%]
  160. unzip -d prep_[% lang %] [% xpi %]
  161. cp $rootdir/bookmarks.html prep_[% lang %]/browser/chrome/[% lang %]/locale/browser/
  162. rm [% xpi %]
  163. cd prep_[% lang %]
  164. [% c('zip', {
  165. zip_src => [ '.' ],
  166. zip_args => xpi,
  167. }) %]
  168. [% END -%]
  169. # If we are building a multi-lingual package, add all of the language packs.
  170. [% IF c("var/multi_lingual") %]
  171. cp [% xpi %] "$TBDIR/$EXTSPATH/langpack-[% lang %]@firefox.mozilla.org.xpi"
  172. [% END %]
  173. cd ..
  174. rm -rf prep_[% lang %]
  175. [% END %]
  176. [% END %]
  177. [% IF c("var/multi_lingual") %]
  178. # Set the update.locale (it is used to replace %LOCALE% within
  179. # app.update.url).
  180. pushd "$TBDIR[% IF c("var/osx") %]/Contents/Resources/[% END %]"
  181. echo ${PKG_LOCALE} > update.locale
  182. [% c("var/touch") %] update.locale
  183. zip -Xm omni.ja update.locale
  184. popd
  185. [% END %]
  186. [% IF c("var/windows") %]
  187. # We need to install a recent version of python-pefile so that it works
  188. # in a x86_64 container:
  189. # https://github.com/TheTorProject/tor-messenger-build/pull/10
  190. tar xf $rootdir/[% c('input_files_by_name/python-pefile') %]
  191. cd $(echo [% c('input_files_by_name/python-pefile') %] | sed s/\.tar\.gz$//)
  192. python setup.py install --user
  193. cd ..
  194. tar -C /var/tmp/dist -xf $rootdir/[% c('input_files_by_name/nsis') %]
  195. export PATH="/var/tmp/dist/nsis/bin:$PATH"
  196. tar -C $distdir -xf $rootdir/[% c('input_files_by_name/tbb-windows-installer') %]
  197. mv ${TB_STAGE_DIR} $distdir/tbb-windows-installer/"Tor Browser"
  198. mv $distdir/tbb-windows-installer ${TB_STAGE_DIR}
  199. [% END %]
  200. [% IF c("var/windows") %]
  201. TBDIR="$distdir/$PKG_DIR/Tor Browser/Browser"
  202. [% ELSIF c("var/osx") %]
  203. TBDIR="$distdir/$PKG_DIR/Tor Browser.app"
  204. [% ELSE %]
  205. TBDIR="$distdir/$PKG_DIR/Browser"
  206. [% END %]
  207. cat > "$scripts_dir/create-$PKG_DIR" << SCRIPT_EOF
  208. #!/bin/bash
  209. set -e
  210. cp -a ${TB_STAGE_DIR} $distdir/$PKG_DIR
  211. pushd "$TBDIR[% IF c("var/osx") %]/Contents/Resources/[% END %]"
  212. rm -f precomplete
  213. python $MARTOOLS/createprecomplete.py
  214. popd
  215. cd $distdir
  216. [% IF c("var/build_mar") -%]
  217. # Create full MAR file and compressed package.
  218. [% SET mar_file = 'tor-browser-' _ c("var/mar_osname") _ '-' _ c("var/torbrowser_version") _ '_${PKG_LOCALE}.mar' %]
  219. MAR=$MARTOOLS/mar \
  220. MOZ_PRODUCT_VERSION=[% c("var/torbrowser_version") %] \
  221. MAR_CHANNEL_ID=torbrowser-torproject-[% c("var/channel") %] \
  222. $MARTOOLS/make_full_update.sh -q $OUTDIR/[% mar_file %] "$TBDIR"
  223. [% END -%]
  224. [% IF c("var/linux") %]
  225. [% c('tar', {
  226. tar_src => [ '$PKG_DIR' ],
  227. tar_args => '-cJf $OUTDIR/tor-browser-' _ c("var/mar_osname") _ '-' _ c("var/torbrowser_version") _ '_${PKG_LOCALE}.tar.xz',
  228. }) %]
  229. [% ELSIF c("var/osx") %]
  230. [% c('var/ddmg', {
  231. dmg_src => '"$PKG_DIR"',
  232. dmg_out => '$OUTDIR/TorBrowser-' _ c("var/torbrowser_version") _ '-osx64_${PKG_LOCALE}.dmg',
  233. }) %]
  234. [% ELSIF c("var/windows") %]
  235. find "$PKG_DIR" -exec [% c("var/touch") %] {} \;
  236. pushd "$PKG_DIR"
  237. makensis torbrowser.nsi
  238. # Working around NSIS braindamage
  239. mv torbrowser-install.exe torbrowser-install-tmp.exe
  240. python $rootdir/pe_checksum_fix.py
  241. mv torbrowser-install-tmp2.exe torbrowser-install.exe
  242. rm torbrowser-install-tmp.exe
  243. mv torbrowser-install.exe $OUTDIR/torbrowser-install[% IF c("var/windows-x86_64") %]-win64[% END %]-[% c("var/torbrowser_version") %]_${PKG_LOCALE}.exe
  244. popd
  245. [% END %]
  246. rm -rf $distdir/${PKG_DIR}
  247. SCRIPT_EOF
  248. cp $rootdir/[% c('input_files_by_name/firefox') %]/mar-tools-*.zip "$OUTDIR"/
  249. [% IF c("var/linux-x86_64") -%]
  250. cp $rootdir/[% c('input_files_by_name/firefox') %]/tor-browser-debug.tar.xz "$OUTDIR"/tor-browser-[% c("var/mar_osname") %]-debug.tar.xz
  251. [% IF !c("var/asan") -%]
  252. cp $rootdir/[% c('input_files_by_name/firefox') %]/geckodriver-linux64.tar.xz "$OUTDIR"/
  253. [% END -%]
  254. [% END -%]
  255. [% IF c("var/linux") -%]
  256. cp $rootdir/[% c('input_files_by_name/tor') %]/tor-debug.tar.xz "$OUTDIR"/tor-[% c("var/mar_osname") %]-debug.tar.xz
  257. [% END -%]
  258. [% IF c("var/build_infos_json") -%]
  259. cp $rootdir/[% c('input_files_by_name/firefox') %]/build-infos.json "$OUTDIR"/build-infos-[% c("var/mar_osname") %].json
  260. [% END -%]
  261. # Create a tarball with all Linux x86_64 language packs (Bug 32676)
  262. [% IF c("var/linux-x86_64") && ! c("var/multi_lingual") && ! c("var/testbuild") -%]
  263. pushd $rootdir/[% c('input_files_by_name/firefox-langpacks') %]
  264. [% c('tar', {
  265. tar_src => [ '.' ],
  266. tar_args => '-cJf $OUTDIR/langpacks-tor-browser-linux64-' _ c("var/torbrowser_version") _ '.tar.xz',
  267. })
  268. %]
  269. popd
  270. [% END -%]
  271. # If we did not create a multi-lingual package above, create a package for
  272. # each locale.
  273. [% IF ! c("var/multi_lingual") %]
  274. [% SET locales = c("var/testbuild") ? [] : c("var/locales") -%]
  275. [% FOREACH lang = locales %]
  276. [% SET lang = tmpl(lang);
  277. SET xpi = '$rootdir/' _ c('input_files_by_name/firefox-langpacks') _ '/' _ lang _ '.xpi';
  278. SET tbdir = '$distdir/tor-browser_' _ lang;
  279. SET mar_file = 'tor-browser-' _ c("var/mar_osname") _ '-' _ c("var/torbrowser_version") _ '_' _ lang _ '.mar';
  280. IF c("var/osx");
  281. SET browserdir = tbdir _ '/Tor Browser.app';
  282. ELSIF c("var/windows");
  283. SET browserdir = tbdir _ '/Tor Browser/Browser';
  284. ELSE;
  285. SET browserdir = tbdir _ '/Browser';
  286. END;
  287. SET build_mar = c("var/build_mar");
  288. # On nightly we only build mar files for some locales
  289. IF c("var/nightly") && c("var/build_mar");
  290. build_mar = 0;
  291. FOREACH mar_lang = c("var/mar_locales");
  292. mar_lang = tmpl(mar_lang);
  293. IF mar_lang == lang;
  294. build_mar = 1;
  295. END;
  296. END;
  297. END;
  298. %]
  299. cat > "$scripts_dir/create-tor-browser_[% lang %]" << SCRIPT_EOF
  300. #!/bin/bash
  301. set -e
  302. cp -a ${TB_STAGE_DIR} [% tbdir %]
  303. cp [% xpi %] "[% browserdir %]/$EXTSPATH/langpack-[% lang %]@firefox.mozilla.org.xpi"
  304. pushd "[% browserdir %]/[% IF c("var/osx") %]Contents/Resources/[% END %]browser/"
  305. mkdir -p defaults/preferences
  306. cp $rootdir/000-tor-browser.js defaults/preferences/
  307. # Set the locale of the bundle.
  308. echo "pref(\"intl.locale.requested\", \"[% lang %]\");" >> defaults/preferences/000-tor-browser.js
  309. [% c("var/touch") %] defaults/preferences/000-tor-browser.js
  310. zip -Xm omni.ja defaults/preferences/000-tor-browser.js
  311. rm -rf defaults
  312. # create tbb_version.json file for #25020
  313. echo '{"version":"[% c("var/torbrowser_version") %]","architecture":"[% c("var/mar_osname") %]","channel":"[% c("var/channel") %]","locale":"[% lang %]"}' > ../tbb_version.json
  314. popd
  315. # Set the update.locale (it is used to replace %LOCALE% within
  316. # app.update.url), remove the en-US spellchecking dictionary, and
  317. # recreate precomplete file (needs to be accurate for full MAR updates).
  318. pushd "[% browserdir %]/[% IF c("var/osx") %]Contents/Resources/[% END %]"
  319. echo "[% lang %]" > update.locale
  320. [% c("var/touch") %] update.locale
  321. zip -Xm omni.ja update.locale
  322. rm -rf dictionaries
  323. rm -f precomplete
  324. python $MARTOOLS/createprecomplete.py
  325. popd
  326. cd $distdir
  327. [% IF build_mar -%]
  328. # Create full MAR file and compressed package for this locale.
  329. MAR=$MARTOOLS/mar \
  330. MOZ_PRODUCT_VERSION=[% c("var/torbrowser_version") %] \
  331. MAR_CHANNEL_ID=torbrowser-torproject-[% c("var/channel") %] \
  332. $MARTOOLS/make_full_update.sh -q $OUTDIR/[% mar_file %] "[% browserdir %]"
  333. [% END -%]
  334. [% IF c("var/linux") %]
  335. [% SET tardir = 'tor-browser_' _ lang;
  336. c('tar', {
  337. tar_src => [ tardir ],
  338. tar_args => '-cJf $OUTDIR/tor-browser-' _ c("var/mar_osname") _ '-' _ c("var/torbrowser_version") _ '_' _ lang _ '.tar.xz',
  339. }) %]
  340. [% ELSIF c("var/osx") %]
  341. [% # Rename the Japanese bundle to not confuse users
  342. IF lang == 'ja-JP-mac';
  343. SET lang = 'ja';
  344. END; -%]
  345. [% c('var/ddmg', {
  346. dmg_src => tbdir,
  347. dmg_out => '$OUTDIR/TorBrowser-' _ c("var/torbrowser_version") _ '-osx64_' _ lang _ '.dmg',
  348. }) %]
  349. [% ELSIF c("var/windows") %]
  350. find "[% tbdir %]" -exec [% c("var/touch") %] {} \;
  351. pushd "[% tbdir %]"
  352. makensis torbrowser.nsi
  353. # Working around NSIS braindamage
  354. mv torbrowser-install.exe torbrowser-install-tmp.exe
  355. python $rootdir/pe_checksum_fix.py
  356. mv torbrowser-install-tmp2.exe torbrowser-install.exe
  357. rm torbrowser-install-tmp.exe
  358. mv torbrowser-install.exe $OUTDIR/torbrowser-install[% IF c("var/windows-x86_64") %]-win64[% END %]-[% c("var/torbrowser_version") %]_[% lang %].exe
  359. popd
  360. [% END %]
  361. rm -rf [% tbdir %]
  362. SCRIPT_EOF
  363. [% END %]
  364. [% END %]
  365. chmod 775 $rootdir/run_scripts "$scripts_dir"/*
  366. $rootdir/run_scripts [% c("buildconf/num_procs") %] "$scripts_dir"