diff --git a/content/sonar/widgets/project-coverage/content.html b/content/sonar/widgets/project-coverage/content.html
new file mode 100644
index 0000000..4a1b2de
--- /dev/null
+++ b/content/sonar/widgets/project-coverage/content.html
@@ -0,0 +1,19 @@
+
+
\ No newline at end of file
diff --git a/content/sonar/widgets/project-coverage/description.yml b/content/sonar/widgets/project-coverage/description.yml
new file mode 100644
index 0000000..0753aaf
--- /dev/null
+++ b/content/sonar/widgets/project-coverage/description.yml
@@ -0,0 +1,4 @@
+name: Metrics
+description: Display the selected Sonar metrics of a new code in an application
+technicalName: sonar
+delay: 500
\ No newline at end of file
diff --git a/content/sonar/widgets/project-coverage/image.png b/content/sonar/widgets/project-coverage/image.png
new file mode 100644
index 0000000..f1faba0
Binary files /dev/null and b/content/sonar/widgets/project-coverage/image.png differ
diff --git a/content/sonar/widgets/project-coverage/params.yml b/content/sonar/widgets/project-coverage/params.yml
new file mode 100644
index 0000000..0188064
--- /dev/null
+++ b/content/sonar/widgets/project-coverage/params.yml
@@ -0,0 +1,49 @@
+widgetParams:
+ -
+ name: 'SURI_TITLE'
+ description: 'Widget title'
+ type: TEXT
+ usageExample: 'My title'
+ required: true
+ -
+ name: 'SURI_PROJECT_KEY'
+ description: 'Project key'
+ type: TEXT
+ usageExample: 'com.project.key:ProjectName'
+ required: true
+ -
+ name: 'SURI_METRICS'
+ description: 'Metrics to display'
+ type: MULTIPLE
+ possibleValuesMap:
+ -
+ jsKey: 'new_line_coverage'
+ value: 'Lines of code'
+ -
+ jsKey: 'new_lines_to_cover'
+ value: 'Lines to Cover'
+ -
+ jsKey: 'new_uncovered_lines'
+ value: 'Uncovered Lines'
+ -
+ jsKey: 'new_coverage'
+ value: 'Coverage on New Code(%)'
+ -
+ jsKey: 'new_conditions_to_cover'
+ value: 'Conditions to Cover'
+ -
+ jsKey: 'new_uncovered_conditions'
+ value: 'Unconver conditions'
+ required: true
+ -
+ name: 'SURI_BRANCH'
+ description: 'Branch name, if user provides both branch and pull request, the branch takes precedence.'
+ type: TEXT
+ usageExample: 'My branch'
+ required: false
+ -
+ name: 'SURI_PULL_REQUEST'
+ description: 'Merge/Pull Request Number'
+ type: TEXT
+ usageExample: '111'
+ required: false
diff --git a/content/sonar/widgets/project-coverage/script.js b/content/sonar/widgets/project-coverage/script.js
new file mode 100644
index 0000000..0eec60e
--- /dev/null
+++ b/content/sonar/widgets/project-coverage/script.js
@@ -0,0 +1,48 @@
+/*
+ * Copyright 2012-2021 the original author or authors.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+function run() {
+ var data = {};
+ var branch = (SURI_BRANCH != null ? "branch="
+ + SURI_BRANCH + "&" : (SURI_PULL_REQUEST != null ? "pullRequest=" + SURI_PULL_REQUEST + "&" : ""));
+ data.results = [];
+
+ // added to remove the trailing slash from the URL if present
+ data.sonarConfigUrl = (WIDGET_CONFIG_SONAR_URL) ? WIDGET_CONFIG_SONAR_URL.replace(/\/+$/, '') : WIDGET_CONFIG_SONAR_URL;
+
+ var response = JSON.parse(
+ Packages.get(data.sonarConfigUrl + "/api/measures/component?" + branch
+ +"component=" + SURI_PROJECT_KEY + "&additionalFields=metrics&metricKeys=" + SURI_METRICS,
+ "Authorization", "Basic " + Packages.btoa(WIDGET_CONFIG_SONAR_TOKEN + ":")));
+
+
+ if (response && response.component && response.component.measures && response.component.measures.length > 0) {
+ response.component.measures.forEach(function(measure) {
+ data.results.push({
+ title: response.metrics.filter(function(metric) {
+ if (metric.key == measure.metric) {
+ return metric;
+ }
+ })[0].name,
+ value:measure.period.value,
+
+ });
+ });
+
+ }
+
+ return JSON.stringify(data);
+}
diff --git a/content/sonar/widgets/project-coverage/style.css b/content/sonar/widgets/project-coverage/style.css
new file mode 100644
index 0000000..1be4ad4
--- /dev/null
+++ b/content/sonar/widgets/project-coverage/style.css
@@ -0,0 +1,57 @@
+.widget.sonar {
+ background-color: #862651;
+}
+
+.widget.sonar .title {
+ color: rgba(255, 255, 255, 0.7);
+ font-style : bold;
+}
+
+.widget.sonar ul {
+ margin: 0 15px;
+ text-align: left;
+ color: rgba(255, 255, 255, 0.7);
+ list-style: none;
+ padding: 0;
+}
+
+.widget.sonar ul li {
+ margin-bottom: 5px;
+ overflow: hidden;
+}
+
+.widget.sonar ul li .label {
+ color: rgba(255, 255, 255, 0.7);
+}
+
+.widget.sonar ul li .value {
+ float: right;
+ margin-left: 12px;
+ font-weight: 600;
+ color: #fff;
+ font-style:bold;
+
+}
+
+.widget.sonar ul li .value .red {
+ color: #e3394f;
+}
+
+.widget.sonar ul li .value .green {
+ color: #86d751;
+}
+
+.widget.sonar ul li .value i {
+ margin-right: 4px;
+}
+
+.widget.sonar .sonarqube {
+ background: transparent url('') no-repeat left bottom;
+ background-size: contain;
+ opacity: 0.2;
+ height: 25%;
+ width: 25%;
+ position: absolute;
+ bottom: 5%;
+ left: 5%;
+}
\ No newline at end of file
diff --git a/content/sonar/widgets/project-metrics/params.yml b/content/sonar/widgets/project-metrics/params.yml
index cfe999b..1c31724 100644
--- a/content/sonar/widgets/project-metrics/params.yml
+++ b/content/sonar/widgets/project-metrics/params.yml
@@ -31,6 +31,9 @@ widgetParams:
-
jsKey: 'sqale_index'
value: 'Technical debt (total effort (in hours) to fix all the issues)'
+ -
+ jsKey: 'coverage'
+ value: 'Coverage (%)'
-
jsKey: 'blocker_violations'
value: 'Number of issues with blocker severity'
diff --git a/suricate-widget-tester.jar b/suricate-widget-tester.jar
new file mode 100644
index 0000000..49e410f
--- /dev/null
+++ b/suricate-widget-tester.jar
@@ -0,0 +1,132422 @@
+#!/bin/bash
+#
+# . ____ _ __ _ _
+# /\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
+# ( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
+# \\/ ___)| |_)| | | | | || (_| | ) ) ) )
+# ' |____| .__|_| |_|_| |_\__, | / / / /
+# =========|_|==============|___/=/_/_/_/
+# :: Spring Boot Startup Script ::
+#
+
+### BEGIN INIT INFO
+# Provides: suricate-widget-tester
+# Required-Start: $remote_fs $syslog $network
+# Required-Stop: $remote_fs $syslog $network
+# Default-Start: 2 3 4 5
+# Default-Stop: 0 1 6
+# Short-Description: suricate widget tester
+# Description: Widget tester of the Suricate application
+# chkconfig: 2345 99 01
+### END INIT INFO
+
+[[ -n "$DEBUG" ]] && set -x
+
+# Initialize variables that cannot be provided by a .conf file
+WORKING_DIR="$(pwd)"
+# shellcheck disable=SC2153
+[[ -n "$JARFILE" ]] && jarfile="$JARFILE"
+[[ -n "$APP_NAME" ]] && identity="$APP_NAME"
+
+# Follow symlinks to find the real jar and detect init.d script
+cd "$(dirname "$0")" || exit 1
+[[ -z "$jarfile" ]] && jarfile=$(pwd)/$(basename "$0")
+while [[ -L "$jarfile" ]]; do
+ if [[ "$jarfile" =~ init\.d ]]; then
+ init_script=$(basename "$jarfile")
+ else
+ configfile="${jarfile%.*}.conf"
+ # shellcheck source=/dev/null
+ [[ -r ${configfile} ]] && source "${configfile}"
+ fi
+ jarfile=$(readlink "$jarfile")
+ cd "$(dirname "$jarfile")" || exit 1
+ jarfile=$(pwd)/$(basename "$jarfile")
+done
+jarfolder="$( (cd "$(dirname "$jarfile")" && pwd -P) )"
+cd "$WORKING_DIR" || exit 1
+
+# Inline script specified in build properties
+
+
+# Source any config file
+configfile="$(basename "${jarfile%.*}.conf")"
+
+# Initialize CONF_FOLDER location defaulting to jarfolder
+[[ -z "$CONF_FOLDER" ]] && CONF_FOLDER="${jarfolder}"
+# shellcheck source=/dev/null
+[[ -r "${CONF_FOLDER}/${configfile}" ]] && source "${CONF_FOLDER}/${configfile}"
+
+# ANSI Colors
+echoRed() { echo $'\e[0;31m'"$1"$'\e[0m'; }
+echoGreen() { echo $'\e[0;32m'"$1"$'\e[0m'; }
+echoYellow() { echo $'\e[0;33m'"$1"$'\e[0m'; }
+
+# Initialize PID/LOG locations if they weren't provided by the config file
+[[ -z "$PID_FOLDER" ]] && PID_FOLDER="/var/run"
+[[ -z "$LOG_FOLDER" ]] && LOG_FOLDER="/var/log"
+! [[ "$PID_FOLDER" == /* ]] && PID_FOLDER="$(dirname "$jarfile")"/"$PID_FOLDER"
+! [[ "$LOG_FOLDER" == /* ]] && LOG_FOLDER="$(dirname "$jarfile")"/"$LOG_FOLDER"
+! [[ -x "$PID_FOLDER" ]] && echoYellow "PID_FOLDER $PID_FOLDER does not exist. Falling back to /tmp" && PID_FOLDER="/tmp"
+! [[ -x "$LOG_FOLDER" ]] && echoYellow "LOG_FOLDER $LOG_FOLDER does not exist. Falling back to /tmp" && LOG_FOLDER="/tmp"
+
+# Set up defaults
+[[ -z "$MODE" ]] && MODE="auto" # modes are "auto", "service" or "run"
+[[ -z "$USE_START_STOP_DAEMON" ]] && USE_START_STOP_DAEMON="true"
+
+# Create an identity for log/pid files
+if [[ -z "$identity" ]]; then
+ if [[ -n "$init_script" ]]; then
+ identity="${init_script}"
+ else
+ identity=$(basename "${jarfile%.*}")_${jarfolder//\//}
+ fi
+fi
+
+# Initialize log file name if not provided by the config file
+[[ -z "$LOG_FILENAME" ]] && LOG_FILENAME="${identity}.log"
+
+# Initialize stop wait time if not provided by the config file
+[[ -z "$STOP_WAIT_TIME" ]] && STOP_WAIT_TIME="60"
+
+# Utility functions
+checkPermissions() {
+ touch "$pid_file" &> /dev/null || { echoRed "Operation not permitted (cannot access pid file)"; return 4; }
+ touch "$log_file" &> /dev/null || { echoRed "Operation not permitted (cannot access log file)"; return 4; }
+}
+
+isRunning() {
+ ps -p "$1" &> /dev/null
+}
+
+await_file() {
+ end=$(date +%s)
+ let "end+=10"
+ while [[ ! -s "$1" ]]
+ do
+ now=$(date +%s)
+ if [[ $now -ge $end ]]; then
+ break
+ fi
+ sleep 1
+ done
+}
+
+# Determine the script mode
+action="run"
+if [[ "$MODE" == "auto" && -n "$init_script" ]] || [[ "$MODE" == "service" ]]; then
+ action="$1"
+ shift
+fi
+
+# Build the pid and log filenames
+PID_FOLDER="$PID_FOLDER/${identity}"
+pid_file="$PID_FOLDER/${identity}.pid"
+log_file="$LOG_FOLDER/$LOG_FILENAME"
+
+# Determine the user to run as if we are root
+# shellcheck disable=SC2012
+[[ $(id -u) == "0" ]] && run_user=$(ls -ld "$jarfile" | awk '{print $3}')
+
+# Ensure the user actually exists
+id -u "$run_user" &> /dev/null || unset run_user
+
+# Run as user specified in RUN_AS_USER
+if [[ -n "$RUN_AS_USER" ]]; then
+ if ! [[ "$action" =~ ^(status|run)$ ]]; then
+ id -u "$RUN_AS_USER" || {
+ echoRed "Cannot run as '$RUN_AS_USER': no such user"
+ exit 2
+ }
+ [[ $(id -u) == 0 ]] || {
+ echoRed "Cannot run as '$RUN_AS_USER': current user is not root"
+ exit 4
+ }
+ fi
+ run_user="$RUN_AS_USER"
+fi
+
+# Issue a warning if the application will run as root
+[[ $(id -u ${run_user}) == "0" ]] && { echoYellow "Application is running as root (UID 0). This is considered insecure."; }
+
+# Find Java
+if [[ -n "$JAVA_HOME" ]] && [[ -x "$JAVA_HOME/bin/java" ]]; then
+ javaexe="$JAVA_HOME/bin/java"
+elif type -p java > /dev/null 2>&1; then
+ javaexe=$(type -p java)
+elif [[ -x "/usr/bin/java" ]]; then
+ javaexe="/usr/bin/java"
+else
+ echo "Unable to find Java"
+ exit 1
+fi
+
+arguments=(-Dsun.misc.URLClassPath.disableJarChecking=true $JAVA_OPTS -jar "$jarfile" $RUN_ARGS "$@")
+
+# Action functions
+start() {
+ if [[ -f "$pid_file" ]]; then
+ pid=$(cat "$pid_file")
+ isRunning "$pid" && { echoYellow "Already running [$pid]"; return 0; }
+ fi
+ do_start "$@"
+}
+
+do_start() {
+ working_dir=$(dirname "$jarfile")
+ pushd "$working_dir" > /dev/null
+ if [[ ! -e "$PID_FOLDER" ]]; then
+ mkdir -p "$PID_FOLDER" &> /dev/null
+ if [[ -n "$run_user" ]]; then
+ chown "$run_user" "$PID_FOLDER"
+ fi
+ fi
+ if [[ ! -e "$log_file" ]]; then
+ touch "$log_file" &> /dev/null
+ if [[ -n "$run_user" ]]; then
+ chown "$run_user" "$log_file"
+ fi
+ fi
+ if [[ -n "$run_user" ]]; then
+ checkPermissions || return $?
+ if [ $USE_START_STOP_DAEMON = true ] && type start-stop-daemon > /dev/null 2>&1; then
+ start-stop-daemon --start --quiet \
+ --chuid "$run_user" \
+ --name "$identity" \
+ --make-pidfile --pidfile "$pid_file" \
+ --background --no-close \
+ --startas "$javaexe" \
+ --chdir "$working_dir" \
+ -- "${arguments[@]}" \
+ >> "$log_file" 2>&1
+ await_file "$pid_file"
+ else
+ su -s /bin/sh -c "$javaexe $(printf "\"%s\" " "${arguments[@]}") >> \"$log_file\" 2>&1 & echo \$!" "$run_user" > "$pid_file"
+ fi
+ pid=$(cat "$pid_file")
+ else
+ checkPermissions || return $?
+ "$javaexe" "${arguments[@]}" >> "$log_file" 2>&1 &
+ pid=$!
+ disown $pid
+ echo "$pid" > "$pid_file"
+ fi
+ [[ -z $pid ]] && { echoRed "Failed to start"; return 1; }
+ echoGreen "Started [$pid]"
+}
+
+stop() {
+ working_dir=$(dirname "$jarfile")
+ pushd "$working_dir" > /dev/null
+ [[ -f $pid_file ]] || { echoYellow "Not running (pidfile not found)"; return 0; }
+ pid=$(cat "$pid_file")
+ isRunning "$pid" || { echoYellow "Not running (process ${pid}). Removing stale pid file."; rm -f "$pid_file"; return 0; }
+ do_stop "$pid" "$pid_file"
+}
+
+do_stop() {
+ kill "$1" &> /dev/null || { echoRed "Unable to kill process $1"; return 1; }
+ for ((i = 1; i <= STOP_WAIT_TIME; i++)); do
+ isRunning "$1" || { echoGreen "Stopped [$1]"; rm -f "$2"; return 0; }
+ [[ $i -eq STOP_WAIT_TIME/2 ]] && kill "$1" &> /dev/null
+ sleep 1
+ done
+ echoRed "Unable to kill process $1";
+ return 1;
+}
+
+force_stop() {
+ [[ -f $pid_file ]] || { echoYellow "Not running (pidfile not found)"; return 0; }
+ pid=$(cat "$pid_file")
+ isRunning "$pid" || { echoYellow "Not running (process ${pid}). Removing stale pid file."; rm -f "$pid_file"; return 0; }
+ do_force_stop "$pid" "$pid_file"
+}
+
+do_force_stop() {
+ kill -9 "$1" &> /dev/null || { echoRed "Unable to kill process $1"; return 1; }
+ for ((i = 1; i <= STOP_WAIT_TIME; i++)); do
+ isRunning "$1" || { echoGreen "Stopped [$1]"; rm -f "$2"; return 0; }
+ [[ $i -eq STOP_WAIT_TIME/2 ]] && kill -9 "$1" &> /dev/null
+ sleep 1
+ done
+ echoRed "Unable to kill process $1";
+ return 1;
+}
+
+restart() {
+ stop && start
+}
+
+force_reload() {
+ working_dir=$(dirname "$jarfile")
+ pushd "$working_dir" > /dev/null
+ [[ -f $pid_file ]] || { echoRed "Not running (pidfile not found)"; return 7; }
+ pid=$(cat "$pid_file")
+ rm -f "$pid_file"
+ isRunning "$pid" || { echoRed "Not running (process ${pid} not found)"; return 7; }
+ do_stop "$pid" "$pid_file"
+ do_start
+}
+
+status() {
+ working_dir=$(dirname "$jarfile")
+ pushd "$working_dir" > /dev/null
+ [[ -f "$pid_file" ]] || { echoRed "Not running"; return 3; }
+ pid=$(cat "$pid_file")
+ isRunning "$pid" || { echoRed "Not running (process ${pid} not found)"; return 1; }
+ echoGreen "Running [$pid]"
+ return 0
+}
+
+run() {
+ pushd "$(dirname "$jarfile")" > /dev/null
+ "$javaexe" "${arguments[@]}"
+ result=$?
+ popd > /dev/null
+ return "$result"
+}
+
+# Call the appropriate action function
+case "$action" in
+start)
+ start "$@"; exit $?;;
+stop)
+ stop "$@"; exit $?;;
+force-stop)
+ force_stop "$@"; exit $?;;
+restart)
+ restart "$@"; exit $?;;
+force-reload)
+ force_reload "$@"; exit $?;;
+status)
+ status "$@"; exit $?;;
+run)
+ run "$@"; exit $?;;
+*)
+ echo "Usage: $0 {start|stop|force-stop|restart|force-reload|status|run}"; exit 1;
+esac
+
+exit 0
+PK 5V META-INF/ PK PK 5V META-INF/MANIFEST.MFmn0~/r#H@T
jKN 盙
oIγ5ScS o1l(:d"_.Wr:.`uHYjjQI8hʄWHYD4r|:X1n>ˀj&*@Ֆ4̇3iէNa?ߧ&?kkc\dRur/a{:V̏(O>Ļj6b#Il.I
:*b9v&VOPK]# PK A org/ PK PK A org/springframework/ PK PK A org/springframework/boot/ PK PK A org/springframework/boot/loader/ PK PK A 8 org/springframework/boot/loader/ClassPathIndexFile.classW |ef6M()KZ`iX(封-n0'dff)(j/ X+c#KĭրW5a+xA~-U=E8; wф;X{Xnц;y+#
+&yax/^0bV0Üy
c8G<y8
+>§0.gX|||>f|)G¸w0.e_Ꮿ
+&LhL^ݨ%2fbS&wg3n4&!Vh03jhNVU/u1t1ڽ.gJC;v3M& A~|XiYDfJЬ{D3!k5slr}71L%֭DOV5fsڦ>3rtL\xb5]g RN?i8;6_c>5x-O`gnG])pao4kZw&CRFug%#nG5t'}kR^IOQiiboH9'c;TޑӶ3Jm̝R!4Vf2@V9 )*mCtmwM)n14h(U'YpNE h)ؐ-=Ur(ˑMS{ѢٌZ`EUitTr~~jz[YA3otWA=HQpo416P&l;:| ]{Mة~'*~x?Tՠ>늶ST{D_b_ZoΘHo6jY^WlC>.bs`T@NnT\ö'T OcnBZ_TP;7y17*`bdNr,xS`FFKl^7+5.P}-fcR~ScVJGIN$Y#4lG{gO7k öXoo[Kr@{N#>oX'Kb5UPA){x/5jK~vq,%\ O7tfD^ ȷǪϸ~- I-GVb=DEoЯgHf4J'3Og<> A@ACn$״ r QF<[H|c,>һZl&6zHDv;Q P_D|\ p-3Y@?]Y4
b",!ΒRZ
+h²#%ZrBzk}=!K+$zATG6K-BߋZtKyOyazuwq_NwKIo+ᦨ5"$<}dH+]nO4NhSG+=nކ鏺30Y
_VpF>)j"X3+L>,!EZBcD:Vr,QoPhytx%P|+;ڱ륅JYR+Jkq4{<
,3+!5$Hhy*¤916WO>W>XÈj/ q
+SZVujQU)
e"8:}W54Pv ZM{h\&kIjl*b-dtaa:F FzN9KүchRAnL2Q3 L!g'YGs,`(~N)\G1CZ%އ5S{k2YҴԐ#Q̟K|z4 '4-CŅa?n"oԞuJmUUJCRo?<^oJJ9?Ⱥx\zfqPd.?"+pd'Iw3uz;5[TJ T
+n3墝»\lB6 "{+5oVl"KKi2QBE+