This directory contains different tests for Apache Trafficserver. It is recommended that all test move to this common area under the correct location based on the type of test being added.
The current layout is:
gold_tests/ - contains all the TSQA v4 based tests that run on the Reusable Gold Testing System (AuTest) tools/ - contain programs used to help with testing.
In the future a directory called "unit/" will be added for adding unit tests based on some standardized testing system.
To help with easy running of the tests, there is a autest.sh and bootstrap.py file.
This file is a simple wrapper that will call the AuTest program in a python virtualenv. If the virtualenv is not setup it will try to install system. That will set up the Reusable Gold Testing System on most systems in a Python virtual environment. The wrapper add some basic options to the command to point to the location of the tests. Add --help for more details on options for running autest test system.
This script should try to install python35 or better on the system, and needed python packages for running the tests.
AuTest can be install manually instead of using the wrapper script. The advange of this is that it is often easier to debug issues with the testing system, or the tests. There are two ways this can be done.
- run the bootstrap script then source the path with a "source ./env-test/bin/activate" command. At this point autest command should run without the wrapper script
- The other way is to make sure you install python 3.5 or better on your system. From there install these python packages ( ie pip install ):
When writting for the AuTest system please refer to the current documenation on the online wiki for general use of the system.
Autest allows for the creation of extension to help specilaize and simplify test writting for a given application domian. Minus API addition the extension code will check that python 3.5 or better is used. There is also a new command line argumented added:
--ats-bin < path to bin directory >
This command line argument will point to your build of ATS you want to test. At this time v6.0 or newer of Trafficserver should work.
- name - A name for this instance of ATS
- command - optional argument defining what process to use. Defaults to traffic_server.
- select_ports - have the testing system auto select the ports to use for this instance of ATS
This function will define a sandbox for an instance of trafficserver to run under. The function will return a AuTest process object that will have a number of files and variables define for making it easier to define a test.
The environment of the process will have a number of added environment variables to control trafficserver running the in the sandbox location correctly. This can be used to easily setup other commands that should run under same environment.
# Define default ATS
ts=Test.MakeATSProcess("ts")
# Call traffic_ctrl to set new generation
tr=Test.AddTestRun()
tr.Processes.Default.Command='traffic_ctl'
tr.Processes.Default.ReturnCode=0
# set the environment for traffic_control to run in to be the same as the "ts" ATS instance
tr.Processes.Default.Env=ts.Env
These are the current variable that are define dynamically
port - the ipv4 port to listen on
portv6 - the ipv4 port to listen on
manager_port - the manager port used. This is set even is select_port is False
admin_port - the admin port used. This is set even is select_port is False
A number of file object are define to help with adding values to a given configuration value to for a test, or testing a value exists in a log file. File that are defined currently are:
- squid.log
- error.log
- diags.log
- records.config
- cache.config
- congestion.config
- hosting.config
- ip_allow.config
- log_hosts.config
- logging.config
- metrics.config
- parent.config
- plugin.config
- remap.config
- socks.config
- splitdns.config
- ssl_multicert.config
- storage.config
- volume.config
Create a server
# don't set ports because a config file will set them
ts1 = Test.MakeATSProcess("ts1",select_ports=False)
ts1.Setup.ts.CopyConfig('config/records_8090.config','records.config')
Create a server and get the dynamic port value
# Define default ATS
ts=Test.MakeATSProcess("ts")
#first test is a miss for default
tr=Test.AddTestRun()
# get port for command from Variables
tr.Processes.Default.Command='curl "http://127.0.0.1:{0}" --verbose'.format(ts.Variables.port)
Add value to a configuration file
# setup some config file for this server
ts.Disk.records_config.update({
'proxy.config.body_factory.enable_customizations': 3, # enable domain specific body factory
'proxy.config.http.cache.generation':-1, # Start with cache turned off
'proxy.config.config_update_interval_ms':1,
})
ts.Disk.plugin_config.AddLine('xdebug.so')
ts.Disk.remap_config.AddLines([
'map /default/ http://127.0.0.1/ @plugin=generator.so',
#line 2
'map /generation1/ http://127.0.0.1/' +
' @plugin=conf_remap.so @pparam=proxy.config.http.cache.generation=1' +
' @plugin=generator.so',
#line 3
'map /generation2/ http://127.0.0.1/' +
' @plugin=conf_remap.so @pparam=proxy.config.http.cache.generation=2' +
' @plugin=generator.so'
])
- file - name of the file to copy. Relative paths are relative from the test file location
- targetname - the name name of the file when copied to the correct configuration location
- process - optional process object to use for getting path location to copy to. Only needed if the Setup object call is not in the scope of the process object created with the MakeATSProcess(...) API.
This function copies a given configuration file the location of a given trafficserver sandbox used in a test. Given a test might have more than on trafficserver instance, it can be difficult to understand the correct location to copy to. This function will deal with the details correctly.
Copy a file over
ts1 = Test.MakeATSProcess("ts1",select_ports=False)
# uses the setup object in the scope of the process object
ts1.Setup.ts.CopyConfig('config/records_8090.config','records.config')
ts1 = Test.MakeATSProcess("ts1",select_ports=False)
# uses the Setup in the global process via a variable passing
Test.Setup.ts.CopyConfig('config/records_8090.config','records.config',ts1)
# same as above, but uses the dynamic object model form
Test.Setup.ts.CopyConfig('config/records_8090.config','records.config',Test.Processes.ts1)
- name - A name for this instance of Origin Server.
This function returns a AuTest process object that launches the python-based microserver. Micro-Server is a mock server which responds to client http requests. Microserver needs to be setup for the tests that require an origin server behind ATS. The server reads a JSON-formatted data file that contains request headers and the corresponding response headers. Microserver responds with payload if the response header contains Content-Length or Transfer-Enconding specified.
- filename - name of the file where the request header and response header will be written to in JSON format
- request_header - dictionary of request header
- response_header - dictionary of response header corresponding to the request header.
This function adds the request header and response header to a file which is then read by the microserver to populate request-response map. The key-fields required for the header dictionary are 'headers', 'timestamp' and 'body'.
#create the origin server process
server=Test.MakeOriginServer("server")
#define the request header and the desired response header
request_header={"headers": "GET / HTTP/1.1\r\nHost: www.example.com\r\n\r\n", "timestamp": "1469733493.993", "body": ""}
#desired response form the origin server
response_header={"headers": "HTTP/1.1 200 OK\r\nConnection: close\r\n\r\n", "timestamp": "1469733493.993", "body": ""}
#addResponse adds the transaction to a file which is used by the server
server.addResponse("sessionlog.json", request_header, response_header)
#add remap rule to traffic server
ts.Disk.remap_config.AddLine(
'map http://www.example.com http://127.0.0.1:{0}'.format(server.Variables.Port)
)
- name - A name for this instance of the DNS.
- default - if a list argument is provided, uDNS will reply with the list contents instead of NXDOMAIN if a DNS can't be found for a partcular entry
This function returns a AuTest process object that launches the python-based microDNS (uDNS). uDNS is a mock DNS which responds to DNS queries. uDNS needs to be setup for the tests that require made-up domains. The server reads a JSON-formatted data file that contains mappings of domain to IP addresses. uDNS responds with the approriate IP addresses if the requested domain is in uDNS' mappings.
- addRecords(records=None, jsonFile=None)
This function adds records using either a dictionary, records, or a json file, jsonFile.
The supplied dictionary must be in the form of { 'domain A': [IP1, IP2], 'domain B': [IP3, IP4] }
.
The supplied json file must take the form of
{
"mappings": [
{domain A: [IP1, IP2]},
{domain B: [IP3, IP4]}
]
}
There are 3 ways to utilize uDNS -
Easy way if everything is done on localhost - by adding default option to Test.MakeDNServer:
uDNS by default returns NXDOMAIN for any unknown mappings
# create TrafficServer and uDNS processes
ts = Test.MakeATSProcess("ts")
dns = Test.MakeDNServer("dns", default=['127.0.0.1'])
ts.Disk.records_config.update({
'proxy.config.dns.nameservers': '127.0.0.1:{0}'.format(dns.Variables.Port), # let TrafficServer know where the DNS is located
'proxy.config.dns.resolv_conf': 'NULL',
'proxy.config.url_remap.remap_required': 0 # need this so TrafficServer won't return error upon not finding the domain in its remap file
})
Using the addRecords method:
# create TrafficServer and uDNS processes
ts = Test.MakeATSProcess("ts")
dns = Test.MakeDNServer("dns")
ts.Disk.records_config.update({
'proxy.config.dns.nameservers': '127.0.0.1:{0}'.format(dns.Variables.Port), # let TrafficServer know where the DNS is located
'proxy.config.dns.resolv_conf': 'NULL',
'proxy.config.url_remap.remap_required': 0 # need this so TrafficServer won't return error upon not finding the domain in its remap file
})
dns.addRecords(records={"foo.com.":["127.0.0.1", "127.0.1.1"]})
# AND/OR
dns.addRecords(jsonFile="zone.json") # where zone.json is in the format described above
Without disabling remap_required:
# create TrafficServer and uDNS processes
ts = Test.MakeATSProcess("ts")
dns = Test.MakeDNServer("dns")
ts.Disk.records_config.update({
'proxy.config.dns.resolv_conf': 'NULL',
'proxy.config.dns.nameservers': '127.0.0.1:{0}'.format(dns.Variables.Port) # let TrafficServer know where the DNS is located
})
# if we don't disable remap_required, we can also just remap a domain to a domain recognized by DNS
ts.Disk.remap_config.AddLine(
'map http://example.com http://foo.com'
)
dns.addRecords(records={"foo.com.":["127.0.0.1", "127.0.1.1"]})
- feature - The feature to test for
This function tests for Traffic server for possible feature it has been compiled with. Current Features you can test for are:
- TS_HAS_LIBZ
- TS_HAS_LZMA
- TS_HAS_JEMALLOC
- TS_HAS_TCMALLOC
- TS_HAS_IN6_IS_ADDR_UNSPECIFIED
- TS_HAS_BACKTRACE
- TS_HAS_PROFILER
- TS_USE_FAST_SDK
- TS_USE_DIAGS
- TS_USE_EPOLL
- TS_USE_KQUEUE
- TS_USE_PORT
- TS_USE_POSIX_CAP
- TS_USE_TPROXY
- TS_HAS_SO_MARK
- TS_HAS_IP_TOS
- TS_USE_HWLOC
- TS_USE_TLS_NPN
- TS_USE_TLS_ALPN
- TS_USE_CERT_CB
- TS_USE_SET_RBIO
- TS_USE_TLS_ECKEY
- TS_USE_LINUX_NATIVE_AIO
- TS_HAS_SO_PEERCRED
- TS_USE_REMOTE_UNWINDING
- TS_HAS_128BIT_CAS
- TS_HAS_TESTS
- TS_HAS_WCCP
- SPLIT_DNS
#create the origin server process
Test.SkipUnless(
Condition.HasATSFeature('TS_USE_TLS_ALPN'),
)
- feature - The feature to test for
This function tests for Curl for possible feature it has been compiled with. Consult Curl documenation for feature set.
#create the origin server process
Test.SkipUnless(
Condition.HasCurlFeature('http2'),
)
- pluginname - The plugin to test for
This function tests for existence of a certain plugin in TrafficServer.
Test.SkipUnless(
Condition.PluginExists('a-plugin.so'),
)