manage.sh 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  1. #!/bin/sh
  2. BASE_DIR="$(dirname -- "`readlink -f -- "$0"`")"
  3. cd -- "$BASE_DIR"
  4. set -e
  5. # subshell
  6. PYTHONPATH="$BASE_DIR"
  7. SEARX_DIR="$BASE_DIR/searx"
  8. ACTION="$1"
  9. #
  10. # Python
  11. #
  12. update_packages() {
  13. pip install --upgrade pip
  14. pip install --upgrade setuptools
  15. pip install -Ur "$BASE_DIR/requirements.txt"
  16. }
  17. update_dev_packages() {
  18. update_packages
  19. pip install -Ur "$BASE_DIR/requirements-dev.txt"
  20. }
  21. install_geckodriver() {
  22. echo '[!] Checking geckodriver'
  23. # TODO : check the current geckodriver version
  24. set -e
  25. geckodriver -V > /dev/null 2>&1 || NOTFOUND=1
  26. set +e
  27. if [ -z "$NOTFOUND" ]; then
  28. return
  29. fi
  30. GECKODRIVER_VERSION="v0.24.0"
  31. PLATFORM="`python -c "import six; import platform; six.print_(platform.system().lower(), platform.architecture()[0])"`"
  32. case "$PLATFORM" in
  33. "linux 32bit" | "linux2 32bit") ARCH="linux32";;
  34. "linux 64bit" | "linux2 64bit") ARCH="linux64";;
  35. "windows 32 bit") ARCH="win32";;
  36. "windows 64 bit") ARCH="win64";;
  37. "mac 64bit") ARCH="macos";;
  38. esac
  39. GECKODRIVER_URL="https://github.com/mozilla/geckodriver/releases/download/$GECKODRIVER_VERSION/geckodriver-$GECKODRIVER_VERSION-$ARCH.tar.gz";
  40. if [ -z "$1" ]; then
  41. if [ -z "$VIRTUAL_ENV" ]; then
  42. printf "geckodriver can't be installed because VIRTUAL_ENV is not set, you should download it from\n %s" "$GECKODRIVER_URL"
  43. exit
  44. else
  45. GECKODRIVER_DIR="$VIRTUAL_ENV/bin"
  46. fi
  47. else
  48. GECKODRIVER_DIR="$1"
  49. mkdir -p -- "$GECKODRIVER_DIR"
  50. fi
  51. printf "Installing %s/geckodriver from\n %s" "$GECKODRIVER_DIR" "$GECKODRIVER_URL"
  52. FILE="`mktemp`"
  53. wget -qO "$FILE" -- "$GECKODRIVER_URL" && tar xz -C "$GECKODRIVER_DIR" -f "$FILE" geckodriver
  54. rm -- "$FILE"
  55. chmod 777 -- "$GECKODRIVER_DIR/geckodriver"
  56. }
  57. locales() {
  58. pybabel compile -d "$SEARX_DIR/translations"
  59. }
  60. update_useragents() {
  61. echo '[!] Updating user agent versions'
  62. python utils/fetch_firefox_version.py
  63. }
  64. pep8_check() {
  65. echo '[!] Running pep8 check'
  66. # ignored rules:
  67. # E402 module level import not at top of file
  68. # W503 line break before binary operator
  69. pep8 --exclude=searx/static --max-line-length=120 --ignore "E402,W503" "$SEARX_DIR" "$BASE_DIR/tests"
  70. }
  71. unit_tests() {
  72. echo '[!] Running unit tests'
  73. python -m nose2 -s "$BASE_DIR/tests/unit"
  74. }
  75. py_test_coverage() {
  76. echo '[!] Running python test coverage'
  77. PYTHONPATH="`pwd`" python -m nose2 -C --log-capture --with-coverage --coverage "$SEARX_DIR" -s "$BASE_DIR/tests/unit" \
  78. && coverage report \
  79. && coverage html
  80. }
  81. robot_tests() {
  82. echo '[!] Running robot tests'
  83. PYTHONPATH="`pwd`" python "$SEARX_DIR/testing.py" robot
  84. }
  85. tests() {
  86. set -e
  87. pep8_check
  88. unit_tests
  89. install_geckodriver
  90. robot_tests
  91. set +e
  92. }
  93. #
  94. # Web
  95. #
  96. npm_path_setup() {
  97. which npm || (printf 'Error: npm is not found\n'; exit 1)
  98. export PATH="$(npm bin)":$PATH
  99. }
  100. npm_packages() {
  101. npm_path_setup
  102. echo '[!] install NPM packages'
  103. cd -- "$BASE_DIR"
  104. npm install less@2.7 less-plugin-clean-css grunt-cli
  105. echo '[!] install NPM packages for oscar theme'
  106. cd -- "$BASE_DIR/searx/static/themes/oscar"
  107. npm install
  108. echo '[!] install NPM packages for simple theme'
  109. cd -- "$BASE_DIR/searx/static/themes/simple"
  110. npm install
  111. }
  112. build_style() {
  113. npm_path_setup
  114. lessc --clean-css="--s1 --advanced --compatibility=ie9" "$BASE_DIR/searx/static/$1" "$BASE_DIR/searx/static/$2"
  115. }
  116. styles() {
  117. npm_path_setup
  118. echo '[!] Building legacy style'
  119. build_style themes/legacy/less/style.less themes/legacy/css/style.css
  120. build_style themes/legacy/less/style-rtl.less themes/legacy/css/style-rtl.css
  121. echo '[!] Building courgette style'
  122. build_style themes/courgette/less/style.less themes/courgette/css/style.css
  123. build_style themes/courgette/less/style-rtl.less themes/courgette/css/style-rtl.css
  124. echo '[!] Building pix-art style'
  125. build_style themes/pix-art/less/style.less themes/pix-art/css/style.css
  126. echo '[!] Building bootstrap style'
  127. build_style less/bootstrap/bootstrap.less css/bootstrap.min.css
  128. }
  129. grunt_build() {
  130. npm_path_setup
  131. echo '[!] Grunt build : oscar theme'
  132. grunt --gruntfile "$SEARX_DIR/static/themes/oscar/gruntfile.js"
  133. echo '[!] Grunt build : simple theme'
  134. grunt --gruntfile "$SEARX_DIR/static/themes/simple/gruntfile.js"
  135. }
  136. docker_build() {
  137. # Check if it is a git repository
  138. if [ ! -d .git ]; then
  139. echo "This is not Git repository"
  140. exit 1
  141. fi
  142. if [ ! -x "$(which git)" ]; then
  143. echo "git is not installed"
  144. exit 1
  145. fi
  146. if [ ! git remote get-url origin 2> /dev/null ]; then
  147. echo "there is no remote origin"
  148. exit 1
  149. fi
  150. # This is a git repository
  151. # "git describe" to get the Docker version (for example : v0.15.0-89-g0585788e)
  152. # awk to remove the "v" and the "g"
  153. SEARX_GIT_VERSION=$(git describe --match "v[0-9]*\.[0-9]*\.[0-9]*" HEAD 2>/dev/null | awk -F'-' '{OFS="-"; $1=substr($1, 2); $3=substr($3, 2); print}')
  154. # add the suffix "-dirty" if the repository has uncommited change
  155. git update-index -q --refresh
  156. if [ ! -z "$(git diff-index --name-only HEAD --)" ]; then
  157. SEARX_GIT_VERSION="${SEARX_GIT_VERSION}-dirty"
  158. fi
  159. # Get the last git commit id, will be added to the Searx version (see Dockerfile)
  160. VERSION_GITCOMMIT=$(echo $SEARX_GIT_VERSION | cut -d- -f2-4)
  161. echo "Last commit : $VERSION_GITCOMMIT"
  162. # Check consistency between the git tag and the searx/version.py file
  163. # /!\ HACK : parse Python file with bash /!\
  164. # otherwise it is not possible build the docker image without all Python dependencies ( version.py loads __init__.py )
  165. # SEARX_PYTHON_VERSION=$(python -c "import six; import searx.version; six.print_(searx.version.VERSION_STRING)")
  166. SEARX_PYTHON_VERSION=$(cat searx/version.py | grep "\(VERSION_MAJOR\|VERSION_MINOR\|VERSION_BUILD\) =" | cut -d\= -f2 | sed -e 's/^[[:space:]]*//' | paste -sd "." -)
  167. if [ $(echo "$SEARX_GIT_VERSION" | cut -d- -f1) != "$SEARX_PYTHON_VERSION" ]; then
  168. echo "Inconsistency between the last git tag and the searx/version.py file"
  169. echo "git tag: $SEARX_GIT_VERSION"
  170. echo "searx/version.py: $SEARX_PYTHON_VERSION"
  171. exit 1
  172. fi
  173. # define the docker image name
  174. # /!\ HACK to get the user name /!\
  175. GITHUB_USER=$(git remote get-url origin | sed 's/.*github\.com\/\([^\/]*\).*/\1/')
  176. SEARX_IMAGE_NAME="${GITHUB_USER:-searx}/searx"
  177. # build Docker image
  178. echo "Building image ${SEARX_IMAGE_NAME}:${SEARX_GIT_VERSION}"
  179. sudo docker build \
  180. --build-arg SEARX_GIT_VERSION="${SEARX_GIT_VERSION}" \
  181. --build-arg VERSION_GITCOMMIT="${VERSION_GITCOMMIT}" \
  182. --build-arg LABEL_DATE=$(date -u +"%Y-%m-%dT%H:%M:%SZ") \
  183. --build-arg LABEL_VCS_REF=$(git rev-parse HEAD) \
  184. --build-arg LABEL_VCS_URL=$(git remote get-url origin) \
  185. --build-arg TIMESTAMP_SETTINGS=$(git log -1 --format="%cd" --date=unix -- searx/settings.yml) \
  186. --build-arg TIMESTAMP_UWSGI=$(git log -1 --format="%cd" --date=unix -- dockerfiles/uwsgi.ini) \
  187. -t ${SEARX_IMAGE_NAME}:latest -t ${SEARX_IMAGE_NAME}:${SEARX_GIT_VERSION} .
  188. if [ "$1" = "push" ]; then
  189. sudo docker push ${SEARX_IMAGE_NAME}:latest
  190. sudo docker push ${SEARX_IMAGE_NAME}:${SEARX_GIT_VERSION}
  191. fi
  192. }
  193. #
  194. # Help
  195. #
  196. help() {
  197. [ -z "$1" ] || printf 'Error: %s\n' "$1"
  198. echo "Searx manage.sh help
  199. Commands
  200. ========
  201. help - This text
  202. Build requirements
  203. ------------------
  204. update_packages - Check & update production dependency changes
  205. update_dev_packages - Check & update development and production dependency changes
  206. install_geckodriver - Download & install geckodriver if not already installed (required for robot_tests)
  207. npm_packages - Download & install npm dependencies
  208. update_useragents - Update useragents.json with the most recent versions of Firefox
  209. Build
  210. -----
  211. locales - Compile locales
  212. styles - Build less files
  213. grunt_build - Build files for themes
  214. docker_build - Build Docker image
  215. Tests
  216. -----
  217. unit_tests - Run unit tests
  218. pep8_check - Pep8 validation
  219. robot_tests - Run selenium tests
  220. tests - Run all python tests (pep8, unit, robot_tests)
  221. py_test_coverage - Unit test coverage
  222. "
  223. }
  224. [ "$(command -V "$ACTION" | grep ' function$')" = "" ] \
  225. && help "action not found" \
  226. || "$ACTION" "$2"