Skip to content

Commit 8366853

Browse files
author
Charlie Root
committedJul 28, 2017
convert to locust; Drupal 6
1 parent 93fe93c commit 8366853

13 files changed

+151
-14493
lines changed
 

‎README.md

+5-5
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,15 @@
11
Drupal Loadtest
22
===============
33

4-
Jmeter load test and support scripts for testing Drupal sites.
4+
Locust load test and support scripts for testing Drupal sites.
55

6-
This Jmeter test plan and scripts are used to test out the [Drupal Memcache module](https://drupal.org/project/memcache).
6+
This locust test plan and scripts are used to test out the [Drupal Memcache module](https://drupal.org/project/memcache).
77

88
Assumptions
99
-----------
1010
These scripts assume:
1111
1. A Drupal installation -- webroot and database.
12-
2. [Apache Jmeter](http://jmeter.apache.org/) installed to /usr/local/jmeter/
12+
2. [Locust](http://locust.io/) installed
1313

1414
Running Tests
1515
-------------
@@ -22,6 +22,6 @@ In addition, `preptest.sh` prepares the site for tests by populating it with con
2222
Once that is complete, it will create a database dump in /root so that the same database can be reloaded for subsequent tests.
2323

2424

25-
The `runtest.sh` script restarts services (mysqld, httpd, memcached) to ensure consistency between tests. Then it runs the Jmeter load test (loadtest.jmx), copies test output data to the webroot, and outputs memcached stats from the test run into a summary report.
25+
The `runtest.sh` script restarts services (mysqld, httpd, memcached) to ensure consistency between tests. Then it runs the locust load test, copies test output data to the webroot, and outputs memcached stats from the test run into a summary report.
2626

27-
`runtest.sh` expects one argument, a tag which will be appended to the output directory as a test identifier.
27+
`runtest.sh` requires one argument, a tag which will be appended to the output directory as a test identifier.

‎data/comments.csv

-2,500
This file was deleted.

‎data/gen_comments.sh

-34
This file was deleted.

‎data/gen_nodes.sh

-34
This file was deleted.

‎data/gen_profiles.sh

-34
This file was deleted.

‎data/nodes.csv

-2,500
This file was deleted.

‎data/profiles.csv

-2,500
This file was deleted.

‎data/user.csv

-5,000
This file was deleted.

‎loadtest.jmx

-1,841
This file was deleted.

‎locust_testplan.py

+105
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,105 @@
1+
from locust import HttpLocust, TaskSet, task, events
2+
from bs4 import BeautifulSoup
3+
import random
4+
5+
def is_static_file(file):
6+
if "/misc" in file:
7+
return True
8+
elif "/themes" in file:
9+
return True
10+
else:
11+
return False
12+
13+
def random_word():
14+
import string
15+
length = random.randint(1, 12)
16+
return "".join( [random.choice(string.letters[:26]) for i in xrange(length)] )
17+
18+
def random_sentence():
19+
length = random.randint(3, 15)
20+
return (" ".join(random_word() for i in xrange(length)) + '.').capitalize()
21+
22+
def random_paragraph():
23+
length = random.randint(3, 15)
24+
return (" ".join(random_sentence() for i in xrange(length)))
25+
26+
def fetch_static_assets(session, response):
27+
resource_urls = set()
28+
soup = BeautifulSoup(response.text, "html.parser")
29+
30+
for res in soup.find_all(src=True):
31+
url = res['src']
32+
if is_static_file(url):
33+
resource_urls.add(url)
34+
else:
35+
print "Skipping: " + url
36+
37+
for url in set(resource_urls):
38+
#Note: If you are going to tag different static file paths differently,
39+
#this is where I would normally do that.
40+
session.client.get(url, name="(Static File)")
41+
42+
class AnonBrowsingUser(TaskSet):
43+
@task(15)
44+
def frontpage(l):
45+
response = l.client.get("/", name="(Anonymous) Front page")
46+
fetch_static_assets(l, response)
47+
48+
@task(10)
49+
def nodepage(l):
50+
nid = random.randint(1, 10000)
51+
l.client.get("/node/%i" % nid, name="(Anonymous) /node/[nid]")
52+
53+
@task(3)
54+
def profilepage(l):
55+
uid = random.randint(3, 5000)
56+
l.client.get("/user/%i" % uid, name="(Anonymous) /user/[uid]")
57+
58+
class AuthBrowsingUser(TaskSet):
59+
def on_start(l):
60+
response = l.client.get("/user", name="(Auth) Login")
61+
soup = BeautifulSoup(response.text, "html.parser")
62+
drupal_form_id = soup.select('input[name="form_build_id"]')[0]["value"]
63+
r = l.client.post("/user", {"name":"jeremy", "pass":"12345", "form_id":"user_login", "op":"Log+in", "form_build_id":drupal_form_id}, name="(Auth) Logging in: /user")
64+
65+
@task(15)
66+
def frontpage(l):
67+
response = l.client.get("/", name="(Auth) Front page")
68+
fetch_static_assets(l, response)
69+
70+
@task(10)
71+
def nodepage(l):
72+
nid = random.randint(1, 10000)
73+
l.client.get("/node/%i" % nid, name="(Auth) /node/[nid]")
74+
75+
@task(3)
76+
def profilepage(l):
77+
uid = random.randint(1, 5000)
78+
l.client.get("/user/%i" % uid, name="(Auth) /user/[uid]")
79+
80+
@task(3)
81+
def postcomments(l):
82+
nid = random.randint(1, 10000)
83+
response = l.client.get("/comment/reply/%i" % nid, name="(Auth) Comment form")
84+
soup = BeautifulSoup(response.text, "html.parser")
85+
drupal_form_build_id_object = soup.select('input[name="form_build_id"]')
86+
drupal_form_token_object = soup.select('input[name="form_token"]')
87+
if drupal_form_build_id_object and drupal_form_token_object:
88+
drupal_form_build_id = drupal_form_build_id_object[0]["value"]
89+
drupal_form_token = drupal_form_token_object[0]["value"]
90+
subject = random_sentence()
91+
response = l.client.post("/comment/reply/%i" % nid, {"subject":subject, "comment":random_paragraph(), "form_id":"comment_form", "form_token":drupal_form_token, "op":"Save", "form_build_id":drupal_form_build_id}, name="(Auth) Posting comment", catch_response=True)
92+
if response.status_code != 200:
93+
response.failure("Failed to post comment: " + str(response.status_code))
94+
elif subject not in response.content:
95+
response.failure("Failed to post comment: comment not showing up")
96+
else:
97+
response.success()
98+
99+
class WebsiteAuthUser(HttpLocust):
100+
weight = 1
101+
task_set = AuthBrowsingUser
102+
103+
class WebsiteAnonUser(HttpLocust):
104+
weight = 4
105+
task_set = AnonBrowsingUser

‎scripts/memcache.settings.inc

+4-6
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,7 @@
11

2-
// Memcached settings.
3-
$conf['cache_backends'][] = 'sites/all/modules/memcache/memcache.inc';
2+
// Drupal 6 memcached settings.
3+
$conf['cache_inc'] ='sites/all/modules/memcache/memcache.inc';
44
$conf['lock_inc'] = 'sites/all/modules/memcache/memcache-lock.inc';
55
$conf['memcache_stampede_protection'] = TRUE;
6-
$conf['cache_default_class'] = 'MemCacheDrupal';
7-
8-
// The 'cache_form' bin must be assigned no non-volatile storage.
9-
$conf['cache_class_cache_form'] = 'DrupalDatabaseCache';
6+
// The 'cache_form' bin must be assigned to non-volatile storage.
7+
$conf['memcache_bins']['cache_form'] = 'database';

‎scripts/preptest.sh

+4-4
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,12 @@
1-
#!/bin/bash
1+
#!/usr/bin/env bash
22
#
33
# Prep a base Drupal install for memcache testing.
44
# This assumes that $WEBDIR contains a working Drupal install.
55

66
WEBDIR=/var/www/html
77
MEMCACHE_SETTINGS_FILE=memcache.settings.inc
8-
MEMCACHE_VERSION=7.x-1.2
9-
DATABASE_NAME=drupal
8+
MEMCACHE_VERSION=6.x-1.11
9+
DATABASE_NAME=drupal6loadtest
1010

1111
USER_COUNT=5000
1212
CONTENT_COUNT=10000
@@ -40,7 +40,7 @@ cat ${MEMCACHE_SETTINGS_FILE} >> ${WEBDIR}/sites/default/settings.php
4040
echo "Setting Drupal usernames and passwords to those that the test expects..."
4141
for i in $(seq 2 5001)
4242
do
43-
mysql -e "UPDATE ${DATABASE_NAME}.users SET name='user${i}', pass='\$S\$DDHmqnax8wgiX9XxV5D9if52hnqvvk2O9KWVGDL1UP5mAdwGov8i' WHERE uid = ${i};"
43+
mysql -e "UPDATE ${DATABASE_NAME}.users SET name='user${i}', pass=MD5('12345') WHERE uid = ${i};"
4444
done
4545

4646
echo "Updating permissions: allow anonymous to access user profiles."

‎scripts/runtest.sh

+33-35
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
#!/bin/bash
1+
#!/usr/bin/env bash
22

33
# Pass the script a tag to identify this testrun (e.g. memcache module version being tested)
44
TAG=$1
@@ -9,35 +9,52 @@ then
99
exit 1
1010
fi
1111

12+
DOMAIN="http://loadtest.dev"
13+
14+
LOCUST=/usr/local/bin/locust
15+
LOCUST_PLAN=/root/drupal-loadtest/locust_testplan.py
16+
17+
RESTART_MYSQL="/usr/local/etc/rc.d/mysql-server restart"
18+
RESTART_APACHE="service apache24 restart"
19+
RESTART_MEMCACHED="service memcached restart"
20+
21+
USERS=100
22+
RAMPUP=10
23+
REQUESTS=50000
24+
1225
DATE=`date +%d-%m-%y--%H:%M:%S-$TAG`
13-
BASEDIR="/root/jmeter"
1426
WEBROOT="/var/www/html"
15-
OUTPUT="$BASEDIR/output"
16-
DEST="$WEBROOT/$DATE"
17-
SECONDS=300
18-
IPADDR=$(/sbin/ifconfig eth0 | /bin/grep 'inet addr' | /bin/cut -d':' -f 2 | /bin/cut -d' ' -f 1)
27+
OUTPUT="$WEBROOT/$DATE"
28+
29+
mkdir $OUTPUT
30+
31+
NC="/usr/bin/nc -N"
32+
33+
IPADDR="ConfigureMe"
34+
1935
# This is output by preptest.sh...
2036
SQL_DUMP=/root/drupal_with_test_content.sql.gz
21-
DB_NAME=drupal
37+
DB_NAME=drupal6loadtest
2238

2339
# Load the database into MySQL so each test starts with the same base data.
2440
echo "Reloading DB, this will likely take a few minutes..."
2541
mysql -e "DROP DATABASE $DB_NAME"
2642
mysql -e "CREATE DATABASE $DB_NAME"
2743
gunzip -c $SQL_DUMP | mysql $DB_NAME
2844

29-
/sbin/service mysqld restart
30-
/sbin/service httpd restart
31-
/sbin/service memcached restart
45+
$RESTART_MYSQL 2>&1 > $OUTPUT/mysql_restart.log
46+
$RESTART_APACHE 2>&1 > $OUTPUT/apache_restart.log
47+
$RESTART_MEMCACHED 2>&1 > $OUTPUT/memcached_restart.log
48+
49+
# Run loadtest
50+
$LOCUST -f $LOCUST_PLAN --host=$DOMAIN --no-web -c $USERS -r $RAMPUP -n $REQUESTS --only-summary --logfile=$OUTPUT/locust.txt
3251

33-
/usr/local/jmeter/bin/jmeter -n -t ${BASEDIR}/loadtest.jmx -j $BASEDIR/jmeter.log
34-
mv "$BASEDIR/jmeter.log" $OUTPUT
35-
mv $OUTPUT $DEST
3652
rm -f "$WEBROOT/latest"
37-
ln -s $DEST "$WEBROOT/latest"
53+
ln -s $OUTPUT "$WEBROOT/latest"
54+
3855
# Add .htaccess to override Drupal's default of disabling indexes.
3956
echo "Options +Indexes" > $WEBROOT/latest/.htaccess
40-
echo 'stats' | nc localhost 11211 > "$WEBROOT/latest/memcached.stats.txt"
57+
echo 'stats' | $NC localhost 11211 > "$WEBROOT/latest/memcached.stats.txt"
4158

4259
SUMMARY="$WEBROOT/latest/summary.tsv"
4360

@@ -66,26 +83,7 @@ echo "\"Miss rate\" $RATE%" >> $SUMMARY 2>&1
6683

6784
echo >> $SUMMARY 2>&1
6885

69-
C20x=`grep "rc=\"20" "$WEBROOT/latest/all_queries.jtl" | wc -l` >> $SUMMARY 2>&1
70-
C200=`grep "rc=\"200" "$WEBROOT/latest/all_queries.jtl" | wc -l` >> $SUMMARY 2>&1
71-
C30x=`grep "rc=\"30" "$WEBROOT/latest/all_queries.jtl"| wc -l` >> $SUMMARY 2>&1
72-
C302=`grep "rc=\"302" "$WEBROOT/latest/all_queries.jtl"| wc -l` >> $SUMMARY 2>&1
73-
C40x=`grep "rc=\"40" "$WEBROOT/latest/all_queries.jtl"| wc -l` >> $SUMMARY 2>&1
74-
C50x=`grep "rc=\"50" "$WEBROOT/latest/all_queries.jtl"| wc -l` >> $SUMMARY 2>&1
75-
TOTAL=`expr $C20x + $C30x + $C40x + $C50x` >> $SUMMARY 2>&1
76-
RATE=`echo "scale=2;$TOTAL / $SECONDS" | bc` >> $SUMMARY 2>&1
77-
78-
echo "\"Pages per second\" $RATE" >> $SUMMARY 2>&1
79-
echo >> $SUMMARY 2>&1
80-
81-
echo "\"HTTP status 20x Success\" $C20x" >> $SUMMARY 2>&1
82-
echo "\"HTTP status 30x Redirection\" $C30x" >> $SUMMARY 2>&1
83-
echo "\"HTTP status 40x Client Error\" $C40x" >> $SUMMARY 2>&1
84-
echo "\"HTTP status 50x Server Error\" $C50x" >> $SUMMARY 2>&1
85-
echo >> $SUMMARY 2>&1
86-
87-
echo "\"HTTP status 200 OK\" $C200" >> $SUMMARY 2>&1
88-
echo "\"HTTP status 302 Found\" $C302" >> $SUMMARY 2>&1
86+
echo $OUTPUT/locust.txt >> $SUMMARY 2>&1
8987

9088
echo >> $SUMMARY 2>&1
9189

0 commit comments

Comments
 (0)
Please sign in to comment.