blob: 12e626e978d2b3954319b958996cfd76989da814 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.
*/
package system.redundancy
import com.jayway.restassured.RestAssured
import com.jayway.restassured.config.SSLConfig
import common._
import org.junit.runner.RunWith
import org.scalatest.junit.JUnitRunner
import org.scalatest.{BeforeAndAfterAll, FlatSpec, Matchers}
import spray.json.DefaultJsonProtocol.StringJsonFormat
import spray.json.{pimpAny, _}
/**
* These tests verify that an alarms redundancy (master/slave) configuration
* works as expected. They will only run properly in an environment with two
* alarms containers running concurrently and env var HOST_INDEX set to host0 in
* one container and host1 in the other. This test also assumes that redis and
* the active endpoint authorization are configured. For the auth set the
* ENDPOINT_AUTH env var in your containers to match the testing.auth property
* found in your whisk.properties. To configure redis simply set the REDIS_URL
* env var in your containers to point to the openwhisk redis container and make
* sure the container is deployed. You can run redis.yml to deploy it.
*/
@RunWith(classOf[JUnitRunner])
class AlarmsRedundancyTests
extends FlatSpec
with Matchers
with BeforeAndAfterAll
with WskTestHelpers {
val wskprops = WskProps()
val wsk = new Wsk
var edgeHost = WhiskProperties.getEdgeHost
val auth = WhiskProperties.getBasicAuth
val user = auth.fst
val password = auth.snd
var endpointPrefix = s"https://$user:$password@$edgeHost/alarmstrigger/worker0/"
val defaultAction = Some(TestUtils.getTestActionFilename("hello.js"))
behavior of "Alarms redundancy tests"
it should "fire alarms trigger before the swap" in withAssetCleaner(wskprops) {
(wp, assetHelper) =>
implicit val wskprops = wp // shadow global props and make implicit
val triggerName = s"dummyAlarmsTrigger-${System.currentTimeMillis}"
val ruleName = s"dummyAlarmsRule-${System.currentTimeMillis}"
val actionName = s"dummyAlarmsAction-${System.currentTimeMillis}"
val packageName = "dummyAlarmsPackage"
// the package alarms should be there
val packageGetResult = wsk.pkg.get("/whisk.system/alarms")
println("fetched package alarms")
packageGetResult.stdout should include("ok")
// create package binding
assetHelper.withCleaner(wsk.pkg, packageName) {
(pkg, name) => pkg.bind("/whisk.system/alarms", name)
}
// create action
assetHelper.withCleaner(wsk.action, actionName) { (action, name) =>
action.create(name, defaultAction)
}
// create trigger feed
assetHelper.withCleaner(wsk.trigger, triggerName) {
(trigger, name) =>
trigger.create(name, feed = Some(s"$packageName/alarm"), parameters = Map(
"trigger_payload" -> "alarmTest".toJson,
"cron" -> "* * * * * *".toJson))
}
// create rule
assetHelper.withCleaner(wsk.rule, ruleName) { (rule, name) =>
rule.create(name, trigger = triggerName, action = actionName)
}
println("waiting for triggers")
val activations = wsk.activation.pollFor(N = 3, Some(triggerName)).length
println(s"Found activation size (should be at least 3): $activations")
activations should be >= 3
}
it should "perform active swap by setting host0 active=false" in {
val endpointURL = endpointPrefix + "0/active?active=false"
val expectedResult = "{\"worker\":\"worker0\",\"host\":\"host0\",\"active\":\"swapping\"}".parseJson.asJsObject
makeGetCallWithExpectedResult(endpointURL, expectedResult)
}
it should "verify active swap by checking for host0 active=false" in {
val endpointURL = endpointPrefix + "0/active"
val expectedResult = "{\"worker\":\"worker0\",\"host\":\"host0\",\"active\":false}".parseJson.asJsObject
Thread.sleep(3000)
makeGetCallWithExpectedResult(endpointURL, expectedResult)
}
it should "verify active swap by checking for host1 active=true" in {
val endpointURL = endpointPrefix + "1/active"
val expectedResult = "{\"worker\":\"worker0\",\"host\":\"host1\",\"active\":true}".parseJson.asJsObject
makeGetCallWithExpectedResult(endpointURL, expectedResult)
}
it should "fire alarms trigger again after the swap" in withAssetCleaner(wskprops) {
(wp, assetHelper) =>
implicit val wskprops = wp // shadow global props and make implicit
val triggerName = s"dummyAlarmsTrigger-${System.currentTimeMillis}"
val ruleName = s"dummyAlarmsRule-${System.currentTimeMillis}"
val actionName = s"dummyAlarmsAction-${System.currentTimeMillis}"
val packageName = "dummyAlarmsPackage"
// the package alarms should be there
val packageGetResult = wsk.pkg.get("/whisk.system/alarms")
println("fetched package alarms")
packageGetResult.stdout should include("ok")
// create package binding
assetHelper.withCleaner(wsk.pkg, packageName) {
(pkg, name) => pkg.bind("/whisk.system/alarms", name)
}
// create action
assetHelper.withCleaner(wsk.action, actionName) { (action, name) =>
action.create(name, defaultAction)
}
// create trigger feed
assetHelper.withCleaner(wsk.trigger, triggerName) {
(trigger, name) =>
trigger.create(name, feed = Some(s"$packageName/alarm"), parameters = Map(
"trigger_payload" -> "alarmTest".toJson,
"cron" -> "* * * * * *".toJson))
}
// create rule
assetHelper.withCleaner(wsk.rule, ruleName) { (rule, name) =>
rule.create(name, trigger = triggerName, action = actionName)
}
println("waiting for triggers")
val activations = wsk.activation.pollFor(N = 3, Some(triggerName)).length
println(s"Found activation size (should be at least 3): $activations")
activations should be >= 3
}
private def makeGetCallWithExpectedResult(endpointURL: String, expectedResult: JsObject) = {
val response = RestAssured.
given().
config(RestAssured.config().sslConfig(new SSLConfig().relaxedHTTPSValidation())).
get(endpointURL)
assert(response.statusCode() == 200)
var result = response.body.asString.parseJson.asJsObject
JsObject(result.fields - "hostMachine") shouldBe expectedResult
}
override def afterAll() {
//swap back to original configuration
RestAssured.
given().
config(RestAssured.config().sslConfig(new SSLConfig().relaxedHTTPSValidation())).
get(endpointPrefix + "0/active?active=true")
}
}