-
Notifications
You must be signed in to change notification settings - Fork 62
/
Copy pathstart_test.sh
executable file
·261 lines (217 loc) · 11.3 KB
/
start_test.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
#!/usr/bin/env bash
#=== FUNCTION ================================================================
# NAME: logit
# DESCRIPTION: Log into file and screen.
# PARAMETER - 1 : Level (ERROR, INFO)
# - 2 : Message
#
#===============================================================================
logit()
{
case "$1" in
"INFO")
echo -e " [\e[94m $1 \e[0m] [ $(date '+%d-%m-%y %H:%M:%S') ] $2 \e[0m" ;;
"WARN")
echo -e " [\e[93m $1 \e[0m] [ $(date '+%d-%m-%y %H:%M:%S') ] \e[93m $2 \e[0m " && sleep 2 ;;
"ERROR")
echo -e " [\e[91m $1 \e[0m] [ $(date '+%d-%m-%y %H:%M:%S') ] $2 \e[0m " ;;
esac
}
#=== FUNCTION ================================================================
# NAME: usage
# DESCRIPTION: Helper of the function
# PARAMETER - None
#
#===============================================================================
usage()
{
logit "INFO" "-j <filename.jmx>"
logit "INFO" "-n <namespace>"
logit "INFO" "-c flag to split and copy csv if you use csv in your test"
logit "INFO" "-m flag to copy fragmented jmx present in scenario/project/module if you use include controller and external test fragment"
logit "INFO" "-i <injectorNumber> to scale slaves pods to the desired number of JMeter injectors"
logit "INFO" "-r flag to enable report generation at the end of the test"
exit 1
}
### Parsing the arguments ###
while getopts 'i:mj:hcrn:' option;
do
case $option in
n ) namespace=${OPTARG} ;;
c ) csv=1 ;;
m ) module=1 ;;
r ) enable_report=1 ;;
j ) jmx=${OPTARG} ;;
i ) nb_injectors=${OPTARG} ;;
h ) usage ;;
? ) usage ;;
esac
done
if [ "$#" -eq 0 ]
then
usage
fi
### CHECKING VARS ###
if [ -z "${namespace}" ]; then
logit "ERROR" "Namespace not provided!"
usage
namespace=$(awk '{print $NF}' "${PWD}/namespace_export")
fi
if [ -z "${jmx}" ]; then
#read -rp 'Enter the name of the jmx file ' jmx
logit "ERROR" "jmx jmeter project not provided!"
usage
fi
jmx_dir="${jmx%%.*}"
if [ ! -f "scenario/${jmx_dir}/${jmx}" ]; then
logit "ERROR" "Test script file was not found in scenario/${jmx_dir}/${jmx}"
usage
fi
# Recreating each pods
logit "INFO" "Recreating pod set"
kubectl -n "${namespace}" delete -f k8s/jmeter/jmeter-master.yaml -f k8s/jmeter/jmeter-slave.yaml 2> /dev/null
kubectl -n "${namespace}" apply -f k8s/jmeter
kubectl -n "${namespace}" patch job jmeter-slaves -p '{"spec":{"parallelism":0}}'
logit "INFO" "Waiting for all slaves pods to be terminated before recreating the pod set"
while [[ $(kubectl -n ${namespace} get pods -l jmeter_mode=slave -o 'jsonpath={..status.conditions[?(@.type=="Ready")].status}') != "" ]]; do echo "$(kubectl -n ${namespace} get pods -l jmeter_mode=slave )" && sleep 1; done
# Starting jmeter slave pod
if [ -z "${nb_injectors}" ]; then
logit "WARNING" "Keeping number of injector to 1"
kubectl -n "${namespace}" patch job jmeter-slaves -p '{"spec":{"parallelism":1}}'
else
logit "INFO" "Scaling the number of pods to ${nb_injectors}. "
kubectl -n "${namespace}" patch job jmeter-slaves -p '{"spec":{"parallelism":'${nb_injectors}'}}'
logit "INFO" "Waiting for pods to be ready"
end=${nb_injectors}
for ((i=1; i<=end; i++))
do
validation_string=${validation_string}"True"
done
while [[ $(kubectl -n ${namespace} get pods -l jmeter_mode=slave -o 'jsonpath={..status.conditions[?(@.type=="Ready")].status}' | sed 's/ //g') != "${validation_string}" ]]; do echo "$(kubectl -n ${namespace} get pods -l jmeter_mode=slave )" && sleep 1; done
logit "INFO" "Finish scaling the number of pods."
fi
#Get Master pod details
logit "INFO" "Waiting for master pod to be available"
while [[ $(kubectl -n ${namespace} get pods -l jmeter_mode=master -o 'jsonpath={..status.conditions[?(@.type=="Ready")].status}') != "True" ]]; do echo "$(kubectl -n ${namespace} get pods -l jmeter_mode=master )" && sleep 1; done
master_pod=$(kubectl get pod -n "${namespace}" | grep jmeter-master | awk '{print $1}')
logit "INFO" "master_pod is ${master_pod}"
#Get Slave pod details
slave_pods=($(kubectl get pods -n "${namespace}" | grep jmeter-slave | grep Running | awk '{print $1}'))
slave_num=${#slave_pods[@]}
slave_digit="${#slave_num}"
# jmeter directory in pods
jmeter_directory="/opt/jmeter/apache-jmeter/bin"
# Copying module and config to pods
if [ -n "${module}" ]; then
logit "INFO" "Using modules (test fragments), uploading them in the pods"
module_dir="scenario/module"
logit "INFO" "Number of slaves is ${slave_num}"
logit "INFO" "Processing directory.. ${module_dir}"
for modulePath in $(ls ${module_dir}/*.jmx)
do
module=$(basename "${modulePath}")
for ((i=0; i<end; i++))
do
printf "Copy %s to %s on %s\n" "${module}" "${jmeter_directory}/${module}" "${slave_pods[$i]}"
kubectl -n "${namespace}" cp -c jmslave "${modulePath}" "${slave_pods[$i]}":"${jmeter_directory}/${module}" &
done
kubectl -n "${namespace}" cp -c jmmaster "${modulePath}" "${master_pod}":"${jmeter_directory}/${module}" &
done
logit "INFO" "Finish copying modules in slave pod"
fi
logit "INFO" "Copying ${jmx} to slaves pods"
logit "INFO" "Number of slaves is ${slave_num}"
for ((i=0; i<end; i++))
do
logit "INFO" "Copying scenario/${jmx_dir}/${jmx} to ${slave_pods[$i]}"
kubectl cp -c jmslave "scenario/${jmx_dir}/${jmx}" -n "${namespace}" "${slave_pods[$i]}:/opt/jmeter/apache-jmeter/bin/" &
done # for i in "${slave_pods[@]}"
logit "INFO" "Finish copying scenario in slaves pod"
logit "INFO" "Copying scenario/${jmx_dir}/${jmx} into ${master_pod}"
kubectl cp -c jmmaster "scenario/${jmx_dir}/${jmx}" -n "${namespace}" "${master_pod}:/opt/jmeter/apache-jmeter/bin/" &
logit "INFO" "Installing needed plugins on slave pods"
## Starting slave pod
{
echo "cd ${jmeter_directory}"
echo "sh PluginsManagerCMD.sh install-for-jmx ${jmx} > plugins-install.out 2> plugins-install.err"
echo "jmeter-server -Dserver.rmi.localport=50000 -Dserver_port=1099 -Jserver.rmi.ssl.disable=true >> jmeter-injector.out 2>> jmeter-injector.err &"
echo "trap 'kill -10 1' EXIT INT TERM"
echo "java -jar /opt/jmeter/apache-jmeter/lib/jolokia-java-agent.jar start JMeter >> jmeter-injector.out 2>> jmeter-injector.err"
echo "wait"
} > "scenario/${jmx_dir}/jmeter_injector_start.sh"
# Copying dataset on slave pods
if [ -n "${csv}" ]; then
logit "INFO" "Splitting and uploading csv to pods"
dataset_dir=./scenario/dataset
for csvfilefull in $(ls ${dataset_dir}/*.csv)
do
logit "INFO" "csvfilefull=${csvfilefull}"
csvfile="${csvfilefull##*/}"
logit "INFO" "Processing file.. $csvfile"
lines_total=$(cat "${csvfilefull}" | wc -l)
logit "INFO" "split --suffix-length=\"${slave_digit}\" -d -l $((lines_total/slave_num)) \"${csvfilefull}\" \"${csvfilefull}/\""
split --suffix-length="${slave_digit}" -d -l $((lines_total/slave_num)) "${csvfilefull}" "${csvfilefull}"
for ((i=0; i<end; i++))
do
if [ ${slave_digit} -eq 2 ] && [ ${i} -lt 10 ]; then
j=0${i}
elif [ ${slave_digit} -eq 2 ] && [ ${i} -ge 10 ]; then
j=${i}
elif [ ${slave_digit} -eq 3 ] && [ ${i} -lt 10 ]; then
j=00${i}
elif [ ${slave_digit} -eq 3 ] && [ ${i} -ge 10 ]; then
j=0${i}
elif [ ${slave_digit} -eq 3 ] && [ ${i} -ge 100 ]; then
j=${i}
else
j=${i}
fi
printf "Copy %s to %s on %s\n" "${csvfilefull}${j}" "${csvfile}" "${slave_pods[$i]}"
kubectl -n "${namespace}" cp -c jmslave "${csvfilefull}${j}" "${slave_pods[$i]}":"${jmeter_directory}/${csvfile}" &
done
done
fi
wait
for ((i=0; i<end; i++))
do
logit "INFO" "Starting jmeter server on ${slave_pods[$i]} in parallel"
kubectl cp -c jmslave "scenario/${jmx_dir}/jmeter_injector_start.sh" -n "${namespace}" "${slave_pods[$i]}:/opt/jmeter/jmeter_injector_start"
kubectl exec -c jmslave -i -n "${namespace}" "${slave_pods[$i]}" -- //bin/bash "/opt/jmeter/jmeter_injector_start" &
done
slave_list=$(kubectl -n ${namespace} describe endpoints jmeter-slaves-svc | grep ' Addresses' | awk -F" " '{print $2}')
logit "INFO" "JMeter slave list : ${slave_list}"
slave_array=($(echo ${slave_list} | sed 's/,/ /g'))
## Starting Jmeter load test
source "scenario/${jmx_dir}/.env"
param_host="-Ghost=${host} -Gport=${port} -Gprotocol=${protocol}"
param_test="-GtimeoutConnect=${timeoutConnect} -GtimeoutResponse=${timeoutResponse}"
param_user="-Gthreads=${threads} -Gduration=${duration} -Grampup=${rampup}"
if [ -n "${enable_report}" ]; then
report_command_line="--reportatendofloadtests --reportoutputfolder /report/report-${jmx}-$(date +"%F_%H%M%S")"
fi
echo "slave_array=(${slave_array[@]}); index=${slave_num} && while [ \${index} -gt 0 ]; do for slave in \${slave_array[@]}; do if echo 'test open port' 2>/dev/null > /dev/tcp/\${slave}/1099; then echo \${slave}' ready' && slave_array=(\${slave_array[@]/\${slave}/}); index=\$((index-1)); else echo \${slave}' not ready'; fi; done; echo 'Waiting for slave readiness'; sleep 2; done" > "scenario/${jmx_dir}/load_test.sh"
{
echo "echo \"Installing needed plugins for master\""
echo "cd /opt/jmeter/apache-jmeter/bin"
echo "sh PluginsManagerCMD.sh install-for-jmx ${jmx}"
echo "jmeter ${param_host} ${param_user} ${report_command_line} --logfile /report/${jmx}_$(date +"%F_%H%M%S").jtl --nongui --testfile ${jmx} -Dserver.rmi.ssl.disable=true --remoteexit --remotestart ${slave_list} >> jmeter-master.out 2>> jmeter-master.err &"
echo "trap 'kill -10 1' EXIT INT TERM"
echo "java -jar /opt/jmeter/apache-jmeter/lib/jolokia-java-agent.jar start JMeter >> jmeter-master.out 2>> jmeter-master.err"
echo "wait"
} >> "scenario/${jmx_dir}/load_test.sh"
logit "INFO" "Copying scenario/${jmx_dir}/load_test.sh into ${master_pod}:/opt/jmeter/load_test"
kubectl cp -c jmmaster "scenario/${jmx_dir}/load_test.sh" -n "${namespace}" "${master_pod}:/opt/jmeter/load_test"
logit "INFO" "Starting the performance test"
logit "INFO" "${namespace} ${master_pod}"
kubectl port-forward $(kubectl get pod | grep grafana |awk '{print $1}') 3000 -n ${namespace} &
logit "INFO" "##################################################"
logit "INFO" "You can follow test execution summary on the master pod by running :"
logit "INFO" " kubectl logs -f -c jmmaster -n ${namespace} ${master_pod}"
logit "INFO" "##################################################"
logit "INFO" "Also using Grafana : kubectl ${namespace} port-forward svc/grafana 8443:443"
GRAFANA_LOGIN=$(kubectl -n "${namespace}" get secret grafana-creds -o yaml | grep GF_SECURITY_ADMIN_USER: | awk -F" " '{print $2}' | base64 --decode)
GRAFANA_PASSWORD=$(kubectl -n "${namespace}" get secret grafana-creds -o yaml | grep GF_SECURITY_ADMIN_PASSWORD: | awk -F" " '{print $2}' | base64 --decode)
logit "WARN" "DO NOT SHARE THESES CREDENTIALS"
logit "INFO" " LOGIN : ${GRAFANA_LOGIN}"
logit "INFO" " PASSWORD : ${GRAFANA_PASSWORD}"
logit "INFO" "################################################"