blob: e5654cb0513ea465e404d9d99427c5741e7d5f93 [file] [log] [blame]
/*
* Copyright 2015-2016 IBM Corporation
*
* 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.
*/
package whisk.core.cli.test
import org.junit.runner.RunWith
import org.scalatest.BeforeAndAfterAll
import org.scalatest.junit.JUnitRunner
import common.RunWskAdminCmd
import common.TestHelpers
import common.TestUtils
import common.TestUtils.FORBIDDEN
import common.TestUtils.NOT_FOUND
import common.TestUtils.TIMEOUT
import common.Wsk
import common.WskProps
import common.WskTestHelpers
import spray.json.DefaultJsonProtocol.StringJsonFormat
import spray.json.pimpAny
import whisk.core.entity.Subject
import whisk.core.entity.WhiskPackage
@RunWith(classOf[JUnitRunner])
class WskEntitlementTests
extends TestHelpers
with WskTestHelpers
with BeforeAndAfterAll {
val wsk = new Wsk
lazy val defaultWskProps = WskProps()
lazy val guestWskProps = getAdditionalTestSubject()
override def afterAll() = {
disposeAdditionalTestSubject(guestWskProps.namespace)
}
def getAdditionalTestSubject() = {
val wskadmin = new RunWskAdminCmd {}
val newSubject = Subject().toString
WskProps(
namespace = newSubject,
authKey = wskadmin.cli(Seq("user", "create", newSubject)).stdout.trim)
}
def disposeAdditionalTestSubject(subject: String) = {
val wskadmin = new RunWskAdminCmd {}
withClue(s"failed to delete temporary subject $subject") {
wskadmin.cli(Seq("user", "delete", subject)).stdout should include("Subject deleted")
}
}
val samplePackage = "samplePackage"
val sampleAction = "sampleAction"
val fullSampleActionName = s"$samplePackage/$sampleAction"
val guestNamespace = guestWskProps.namespace
behavior of "Wsk Package Entitlement"
it should "not allow unauthorized subject to operate on private action" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
val privateAction = "privateAction"
assetHelper.withCleaner(wsk.action, privateAction) {
(action, name) => action.create(name, Some(TestUtils.getTestActionFilename("hello.js")))(wp)
}
val fullyQualifiedActionName = s"/$guestNamespace/$privateAction"
wsk.action.get(fullyQualifiedActionName, expectedExitCode = FORBIDDEN)(defaultWskProps).
stderr should include("not authorized")
withAssetCleaner(defaultWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.action, fullyQualifiedActionName, confirmDelete = false) {
(action, name) =>
val rr = action.create(name, None, update = true, expectedExitCode = FORBIDDEN)(wp)
rr.stderr should include("not authorized")
rr
}
assetHelper.withCleaner(wsk.action, "unauthorized sequence", confirmDelete = false) {
(action, name) =>
val rr = action.create(name, Some(fullyQualifiedActionName), kind = Some("sequence"), update = true, expectedExitCode = FORBIDDEN)(wp)
rr.stderr should include("not authorized")
rr
}
}
wsk.action.delete(fullyQualifiedActionName, expectedExitCode = FORBIDDEN)(defaultWskProps).
stderr should include("not authorized")
wsk.action.invoke(fullyQualifiedActionName, expectedExitCode = FORBIDDEN)(defaultWskProps).
stderr should include("not authorized")
}
it should "reject deleting action in shared package not owned by authkey" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) => pkg.create(samplePackage, shared = Some(true))(wp)
}
assetHelper.withCleaner(wsk.action, fullSampleActionName) {
val file = Some(TestUtils.getTestActionFilename("empty.js"))
(action, _) => action.create(fullSampleActionName, file)(wp)
}
val fullyQualifiedActionName = s"/$guestNamespace/$fullSampleActionName"
wsk.action.get(fullyQualifiedActionName)(defaultWskProps)
wsk.action.delete(fullyQualifiedActionName, expectedExitCode = FORBIDDEN)(defaultWskProps)
}
it should "reject create action in shared package not owned by authkey" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, name) => pkg.create(name, shared = Some(true))(wp)
}
val fullyQualifiedActionName = s"/$guestNamespace/notallowed"
val file = Some(TestUtils.getTestActionFilename("empty.js"))
withAssetCleaner(defaultWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.action, fullyQualifiedActionName, confirmDelete = false) {
(action, name) => action.create(name, file, expectedExitCode = FORBIDDEN)(wp)
}
}
}
it should "reject update action in shared package not owned by authkey" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) => pkg.create(samplePackage, shared = Some(true))(wp)
}
assetHelper.withCleaner(wsk.action, fullSampleActionName) {
val file = Some(TestUtils.getTestActionFilename("empty.js"))
(action, _) => action.create(fullSampleActionName, file)(wp)
}
val fullyQualifiedActionName = s"/$guestNamespace/$fullSampleActionName"
wsk.action.create(fullyQualifiedActionName, None, update = true, expectedExitCode = FORBIDDEN)(defaultWskProps)
}
behavior of "Wsk Package Listing"
it should "list shared packages" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) => pkg.create(samplePackage, shared = Some(true))(wp)
}
val fullyQualifiedPackageName = s"/$guestNamespace/$samplePackage"
val result = wsk.pkg.list(Some(s"/$guestNamespace"))(defaultWskProps).stdout
result should include regex (fullyQualifiedPackageName + """\s+shared""")
}
it should "not list private packages" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) => pkg.create(samplePackage)(wp)
}
val fullyQualifiedPackageName = s"/$guestNamespace/$samplePackage"
val result = wsk.pkg.list(Some(s"/$guestNamespace"))(defaultWskProps).stdout
result should not include regex(fullyQualifiedPackageName)
}
it should "list shared package actions" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) => pkg.create(samplePackage, shared = Some(true))(wp)
}
assetHelper.withCleaner(wsk.action, fullSampleActionName) {
val file = Some(TestUtils.getTestActionFilename("empty.js"))
(action, _) => action.create(fullSampleActionName, file, kind = Some("nodejs"))(wp)
}
val fullyQualifiedPackageName = s"/$guestNamespace/$samplePackage"
val fullyQualifiedActionName = s"/$guestNamespace/$fullSampleActionName"
val result = wsk.action.list(Some(fullyQualifiedPackageName))(defaultWskProps).stdout
result should include regex (fullyQualifiedActionName)
}
behavior of "Wsk Package Binding"
it should "create a package binding" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) => pkg.create(samplePackage, shared = Some(true))(wp)
}
val name = "bindPackage"
val annotations = Map("a" -> "A".toJson, WhiskPackage.bindingFieldName -> "xxx".toJson)
val provider = s"/$guestNamespace/$samplePackage"
withAssetCleaner(defaultWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, name) {
(pkg, _) => pkg.bind(provider, name, annotations = annotations)(wp)
}
val stdout = wsk.pkg.get(name)(defaultWskProps).stdout
val annotationString = wsk.parseJsonString(stdout).fields("annotations").toString
annotationString should include regex (""""key":"a"""")
annotationString should include regex (""""value":"A"""")
annotationString should include regex (s""""key":"${WhiskPackage.bindingFieldName}"""")
annotationString should not include regex(""""key":"xxx"""")
annotationString should include regex (s""""name":"${samplePackage}"""")
}
}
it should "not create a package binding for private package" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) => pkg.create(samplePackage, shared = Some(false))(wp)
}
val name = "bindPackage"
val provider = s"/$guestNamespace/$samplePackage"
withAssetCleaner(defaultWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, name, confirmDelete = false) {
(pkg, _) => pkg.bind(provider, name, expectedExitCode = FORBIDDEN)(wp)
}
}
}
behavior of "Wsk Package Action"
it should "get and invoke an action from package" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) => pkg.create(samplePackage, parameters = Map("a" -> "A".toJson), shared = Some(true))(wp)
}
assetHelper.withCleaner(wsk.action, fullSampleActionName) {
val file = Some(TestUtils.getTestActionFilename("hello.js"))
(action, _) => action.create(fullSampleActionName, file)(wp)
}
val fullyQualifiedActionName = s"/$guestNamespace/$fullSampleActionName"
val stdout = wsk.action.get(fullyQualifiedActionName)(defaultWskProps).stdout
stdout should include("name")
stdout should include("parameters")
stdout should include("limits")
stdout should include regex (""""key": "a"""")
stdout should include regex (""""value": "A"""")
val run = wsk.action.invoke(fullyQualifiedActionName)(defaultWskProps)
withActivation(wsk.activation, run)({
_.response.success shouldBe true
})(defaultWskProps)
}
it should "invoke an action sequence from package" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) => pkg.create(samplePackage, parameters = Map("a" -> "A".toJson), shared = Some(true))(wp)
}
assetHelper.withCleaner(wsk.action, fullSampleActionName) {
val file = Some(TestUtils.getTestActionFilename("hello.js"))
(action, _) => action.create(fullSampleActionName, file)(wp)
}
withAssetCleaner(defaultWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.action, "sequence") {
(action, name) =>
val fullyQualifiedActionName = s"/$guestNamespace/$fullSampleActionName"
action.create(name, Some(fullyQualifiedActionName), kind = Some("sequence"), update = true)(wp)
}
val run = wsk.action.invoke("sequence")(defaultWskProps)
withActivation(wsk.activation, run)({
_.response.success shouldBe true
})(defaultWskProps)
}
}
it should "not allow invoke an action sequence with more than one component from package after entitlement change" in withAssetCleaner(guestWskProps) {
(guestwp, assetHelper) =>
val privateSamplePackage = samplePackage + "prv"
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) =>
pkg.create(samplePackage, parameters = Map("a" -> "A".toJson), shared = Some(true))(guestwp)
pkg.create(privateSamplePackage, parameters = Map("a" -> "A".toJson), shared = Some(true))(guestwp)
}
assetHelper.withCleaner(wsk.action, fullSampleActionName) {
val file = Some(TestUtils.getTestActionFilename("hello.js"))
(action, _) =>
action.create(fullSampleActionName, file)(guestwp)
action.create(s"$privateSamplePackage/$sampleAction", file)(guestwp)
}
withAssetCleaner(defaultWskProps) {
(dwp, assetHelper) =>
assetHelper.withCleaner(wsk.action, "sequence") {
(action, name) =>
val fullyQualifiedActionName = s"/$guestNamespace/$fullSampleActionName"
val fullyQualifiedActionName2 = s"/$guestNamespace/$privateSamplePackage/$sampleAction"
action.create(name, Some(s"$fullyQualifiedActionName,$fullyQualifiedActionName2"),
kind = Some("sequence"))(dwp)
}
// change package visibility
wsk.pkg.create(privateSamplePackage, update = true, shared = Some(false))(guestwp)
wsk.action.invoke("sequence", expectedExitCode = FORBIDDEN)(defaultWskProps)
}
}
behavior of "Wsk Trigger Feed"
it should "not create a trigger with timeout error when feed fails to initialize" in withAssetCleaner(guestWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.pkg, samplePackage) {
(pkg, _) => pkg.create(samplePackage, shared = Some(true))(wp)
}
val sampleFeed = s"$samplePackage/sampleFeed"
assetHelper.withCleaner(wsk.action, sampleFeed) {
val file = Some(TestUtils.getTestActionFilename("empty.js"))
(action, _) => action.create(sampleFeed, file, kind = Some("nodejs"))(wp)
}
val fullyQualifiedFeedName = s"/$guestNamespace/$sampleFeed"
withAssetCleaner(defaultWskProps) {
(wp, assetHelper) =>
assetHelper.withCleaner(wsk.trigger, "badfeed", confirmDelete = false) {
(trigger, name) => trigger.create(name, feed = Some(fullyQualifiedFeedName), expectedExitCode = TIMEOUT)(wp)
}
wsk.trigger.get("badfeed", expectedExitCode = NOT_FOUND)(wp)
}
}
}