2020-05-12 13:50:18 +02:00
|
|
|
import com.sap.piper.DefaultValueCache
|
2020-09-24 13:47:20 +02:00
|
|
|
import com.sap.piper.Utils
|
|
|
|
import org.junit.After
|
2017-12-06 13:03:06 +02:00
|
|
|
import org.junit.Before
|
2018-01-16 10:33:13 +02:00
|
|
|
import org.junit.Rule
|
2017-12-06 13:03:06 +02:00
|
|
|
import org.junit.Test
|
2020-05-12 13:50:18 +02:00
|
|
|
import org.junit.rules.ExpectedException
|
2018-01-26 15:55:15 +02:00
|
|
|
import org.junit.rules.RuleChain
|
2017-12-06 13:03:06 +02:00
|
|
|
import org.yaml.snakeyaml.Yaml
|
2018-06-06 11:19:19 +02:00
|
|
|
import util.BasePiperTest
|
2020-05-18 10:59:02 +02:00
|
|
|
import util.JenkinsLoggingRule
|
2020-05-12 13:50:18 +02:00
|
|
|
import util.JenkinsReadFileRule
|
|
|
|
import util.JenkinsShellCallRule
|
2018-02-28 14:11:09 +02:00
|
|
|
import util.JenkinsStepRule
|
2020-03-17 10:19:09 +02:00
|
|
|
import util.JenkinsWriteFileRule
|
2019-07-31 12:22:26 +02:00
|
|
|
import util.Rules
|
2018-01-16 10:33:13 +02:00
|
|
|
|
2020-10-05 12:50:03 +02:00
|
|
|
import static org.hamcrest.Matchers.is
|
2017-12-06 13:03:06 +02:00
|
|
|
import static org.junit.Assert.assertEquals
|
|
|
|
import static org.junit.Assert.assertNotNull
|
2020-09-04 14:45:09 +02:00
|
|
|
import static org.junit.Assert.assertNull
|
2020-10-05 12:50:03 +02:00
|
|
|
import static org.junit.Assert.assertThat
|
2017-12-06 13:03:06 +02:00
|
|
|
|
2018-06-06 11:19:19 +02:00
|
|
|
class SetupCommonPipelineEnvironmentTest extends BasePiperTest {
|
2019-03-26 15:13:03 +02:00
|
|
|
|
2018-03-02 11:57:50 +02:00
|
|
|
def usedConfigFile
|
2020-12-01 13:04:27 +02:00
|
|
|
def pipelineAndTestStashIncludes
|
2018-03-02 11:57:50 +02:00
|
|
|
|
2019-01-22 10:25:42 +02:00
|
|
|
private JenkinsStepRule stepRule = new JenkinsStepRule(this)
|
2020-03-17 10:19:09 +02:00
|
|
|
private JenkinsWriteFileRule writeFileRule = new JenkinsWriteFileRule(this)
|
2020-05-12 13:50:18 +02:00
|
|
|
private ExpectedException thrown = ExpectedException.none()
|
|
|
|
private JenkinsShellCallRule shellRule = new JenkinsShellCallRule(this)
|
|
|
|
private JenkinsReadFileRule readFileRule = new JenkinsReadFileRule(this, "./")
|
2020-05-18 10:59:02 +02:00
|
|
|
private JenkinsLoggingRule loggingRule = new JenkinsLoggingRule(this)
|
2018-01-16 18:06:25 +02:00
|
|
|
|
2018-01-16 10:33:13 +02:00
|
|
|
@Rule
|
2018-02-28 14:11:09 +02:00
|
|
|
public RuleChain rules = Rules
|
|
|
|
.getCommonRules(this)
|
2019-01-22 10:25:42 +02:00
|
|
|
.around(stepRule)
|
2020-03-17 10:19:09 +02:00
|
|
|
.around(writeFileRule)
|
2020-05-12 13:50:18 +02:00
|
|
|
.around(thrown)
|
|
|
|
.around(shellRule)
|
|
|
|
.around(readFileRule)
|
2020-05-18 10:59:02 +02:00
|
|
|
.around(loggingRule)
|
2020-05-12 13:50:18 +02:00
|
|
|
|
2018-02-28 14:11:09 +02:00
|
|
|
|
2017-12-06 13:03:06 +02:00
|
|
|
@Before
|
2018-01-16 10:33:13 +02:00
|
|
|
void init() {
|
2019-03-26 15:13:03 +02:00
|
|
|
|
2017-12-06 13:03:06 +02:00
|
|
|
def examplePipelineConfig = new File('test/resources/test_pipeline_config.yml').text
|
|
|
|
|
2020-05-12 13:50:18 +02:00
|
|
|
helper.registerAllowedMethod("libraryResource", [String], { fileName ->
|
|
|
|
switch(fileName) {
|
|
|
|
case 'default_pipeline_environment.yml': return "default: 'config'"
|
|
|
|
case 'custom.yml': return "custom: 'myConfig'"
|
|
|
|
case 'notFound.yml': throw new hudson.AbortException('No such library resource notFound could be found')
|
|
|
|
default: return "the:'end'"
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2017-12-06 13:03:06 +02:00
|
|
|
helper.registerAllowedMethod("readYaml", [Map], { Map parameters ->
|
|
|
|
Yaml yamlParser = new Yaml()
|
2019-07-31 12:22:26 +02:00
|
|
|
if (parameters.text) {
|
2017-12-06 13:03:06 +02:00
|
|
|
return yamlParser.load(parameters.text)
|
2020-12-01 13:04:27 +02:00
|
|
|
} else if (parameters.file) {
|
|
|
|
switch (parameters.file) {
|
|
|
|
case '.pipeline/default_pipeline_environment.yml':
|
|
|
|
return [default: 'config']
|
|
|
|
case '.pipeline/custom.yml':
|
|
|
|
return [custom: 'myConfig']
|
|
|
|
case 'pipeline_config.yml':
|
|
|
|
usedConfigFile = parameters.file
|
|
|
|
return [
|
|
|
|
general: [
|
|
|
|
productiveBranch: 'main'
|
|
|
|
],
|
|
|
|
steps: [
|
|
|
|
mavenExecute: [
|
|
|
|
dockerImage: 'my-custom-maven-docker']
|
|
|
|
]
|
|
|
|
]
|
|
|
|
}
|
2020-05-12 13:50:18 +02:00
|
|
|
} else {
|
|
|
|
throw new IllegalArgumentException("Key 'text' and 'file' are both missing in map ${m}.")
|
2017-12-06 13:03:06 +02:00
|
|
|
}
|
|
|
|
usedConfigFile = parameters.file
|
|
|
|
return yamlParser.load(examplePipelineConfig)
|
|
|
|
})
|
2020-09-24 13:47:20 +02:00
|
|
|
|
2020-12-01 13:04:27 +02:00
|
|
|
helper.registerAllowedMethod("stash", [Map], { Map params ->
|
|
|
|
pipelineAndTestStashIncludes = params.includes
|
|
|
|
})
|
|
|
|
|
2020-09-24 13:47:20 +02:00
|
|
|
Utils.metaClass.echo = { def m -> }
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@After
|
|
|
|
public void tearDown() {
|
|
|
|
Utils.metaClass = null
|
2017-12-06 13:03:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
2018-09-27 15:45:28 +02:00
|
|
|
void testIsYamlConfigurationAvailable() throws Exception {
|
|
|
|
|
|
|
|
helper.registerAllowedMethod("fileExists", [String], { String path ->
|
|
|
|
return path.endsWith('.pipeline/config.yml')
|
|
|
|
})
|
|
|
|
|
2020-11-11 11:44:49 +02:00
|
|
|
stepRule.step.setupCommonPipelineEnvironment(script: nullScript)
|
2017-12-06 13:03:06 +02:00
|
|
|
|
|
|
|
assertEquals('.pipeline/config.yml', usedConfigFile)
|
2018-06-06 11:19:19 +02:00
|
|
|
assertNotNull(nullScript.commonPipelineEnvironment.configuration)
|
|
|
|
assertEquals('develop', nullScript.commonPipelineEnvironment.configuration.general.productiveBranch)
|
|
|
|
assertEquals('my-maven-docker', nullScript.commonPipelineEnvironment.configuration.steps.mavenExecute.dockerImage)
|
2020-12-01 13:04:27 +02:00
|
|
|
assertEquals('.pipeline/**', pipelineAndTestStashIncludes)
|
2017-12-06 13:03:06 +02:00
|
|
|
}
|
2019-07-31 12:22:26 +02:00
|
|
|
|
|
|
|
@Test
|
|
|
|
void testWorksAlsoWithYamlFileEnding() throws Exception {
|
|
|
|
|
|
|
|
helper.registerAllowedMethod("fileExists", [String], { String path ->
|
|
|
|
return path.endsWith('.pipeline/config.yaml')
|
|
|
|
})
|
|
|
|
|
2020-11-11 11:44:49 +02:00
|
|
|
stepRule.step.setupCommonPipelineEnvironment(script: nullScript)
|
2019-07-31 12:22:26 +02:00
|
|
|
|
|
|
|
assertEquals('.pipeline/config.yaml', usedConfigFile)
|
|
|
|
assertNotNull(nullScript.commonPipelineEnvironment.configuration)
|
|
|
|
assertEquals('develop', nullScript.commonPipelineEnvironment.configuration.general.productiveBranch)
|
|
|
|
assertEquals('my-maven-docker', nullScript.commonPipelineEnvironment.configuration.steps.mavenExecute.dockerImage)
|
2020-12-01 13:04:27 +02:00
|
|
|
assertEquals('.pipeline/**', pipelineAndTestStashIncludes)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
void testWorksAlsoWithCustomConfig() throws Exception {
|
|
|
|
|
|
|
|
helper.registerAllowedMethod("fileExists", [String], { String path ->
|
|
|
|
return path.endsWith('pipeline_config.yml')
|
|
|
|
})
|
|
|
|
|
|
|
|
stepRule.step.setupCommonPipelineEnvironment(script: nullScript, configFile: 'pipeline_config.yml')
|
|
|
|
|
|
|
|
assertEquals('pipeline_config.yml', usedConfigFile)
|
|
|
|
assertNotNull(nullScript.commonPipelineEnvironment.configuration)
|
|
|
|
assertEquals('main', nullScript.commonPipelineEnvironment.configuration.general.productiveBranch)
|
|
|
|
assertEquals('my-custom-maven-docker', nullScript.commonPipelineEnvironment.configuration.steps.mavenExecute.dockerImage)
|
|
|
|
assertEquals('.pipeline/**, pipeline_config.yml', pipelineAndTestStashIncludes)
|
2019-07-31 12:22:26 +02:00
|
|
|
}
|
2020-05-12 13:50:18 +02:00
|
|
|
|
|
|
|
@Test
|
2020-05-18 10:59:02 +02:00
|
|
|
void testAttemptToLoadNonExistingConfigFile() {
|
2020-05-12 13:50:18 +02:00
|
|
|
|
|
|
|
helper.registerAllowedMethod("fileExists", [String], { String path ->
|
|
|
|
switch(path) {
|
|
|
|
case 'default_pipeline_environment.yml': return false
|
|
|
|
case 'custom.yml': return false
|
|
|
|
case 'notFound.yml': return false
|
2020-05-18 10:59:02 +02:00
|
|
|
case '': throw new RuntimeException('cannot call fileExists with empty path')
|
2020-05-12 13:50:18 +02:00
|
|
|
default: return true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
helper.registerAllowedMethod("handlePipelineStepErrors", [Map,Closure], { Map map, Closure closure ->
|
|
|
|
closure()
|
|
|
|
})
|
|
|
|
|
|
|
|
// Behavior documented here based on reality check
|
|
|
|
thrown.expect(hudson.AbortException.class)
|
|
|
|
thrown.expectMessage('No such library resource notFound could be found')
|
|
|
|
|
|
|
|
stepRule.step.setupCommonPipelineEnvironment(
|
|
|
|
script: nullScript,
|
2020-11-11 11:44:49 +02:00
|
|
|
customDefaults: 'notFound.yml'
|
2020-05-12 13:50:18 +02:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-05-18 10:59:02 +02:00
|
|
|
@Test
|
|
|
|
void testInvalidEntriesInCustomDefaults() {
|
|
|
|
|
|
|
|
helper.registerAllowedMethod("fileExists", [String], { String path ->
|
|
|
|
switch(path) {
|
|
|
|
case 'default_pipeline_environment.yml': return false
|
|
|
|
case '': throw new RuntimeException('cannot call fileExists with empty path')
|
|
|
|
default: return true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
helper.registerAllowedMethod("handlePipelineStepErrors", [Map,Closure], { Map map, Closure closure ->
|
|
|
|
closure()
|
|
|
|
})
|
|
|
|
|
|
|
|
helper.registerAllowedMethod("readYaml", [Map], { Map parameters ->
|
|
|
|
Yaml yamlParser = new Yaml()
|
|
|
|
if (parameters.text) {
|
|
|
|
return yamlParser.load(parameters.text)
|
|
|
|
} else if (parameters.file) {
|
|
|
|
if (parameters.file == '.pipeline/config-with-custom-defaults.yml') {
|
|
|
|
return [customDefaults: ['', true]]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
throw new IllegalArgumentException("Unexpected invocation of readYaml step")
|
|
|
|
})
|
|
|
|
|
|
|
|
stepRule.step.setupCommonPipelineEnvironment(
|
|
|
|
script: nullScript,
|
2020-11-11 11:44:49 +02:00
|
|
|
configFile: '.pipeline/config-with-custom-defaults.yml'
|
2020-05-18 10:59:02 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
assertEquals('WARNING: Ignoring invalid entry in custom defaults from files: \'\' \n' +
|
|
|
|
'WARNING: Ignoring invalid entry in custom defaults from files: \'true\' \n', loggingRule.getLog())
|
|
|
|
}
|
|
|
|
|
2020-05-12 13:50:18 +02:00
|
|
|
@Test
|
|
|
|
void testAttemptToLoadFileFromURL() {
|
|
|
|
helper.registerAllowedMethod("fileExists", [String], {String path ->
|
|
|
|
switch (path) {
|
|
|
|
case 'default_pipeline_environment.yml': return false
|
2020-05-18 10:59:02 +02:00
|
|
|
case '': throw new RuntimeException('cannot call fileExists with empty path')
|
2020-05-12 13:50:18 +02:00
|
|
|
default: return true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
String customDefaultUrl = "https://url-to-my-config.com/my-config.yml"
|
|
|
|
boolean urlRequested = false
|
|
|
|
|
|
|
|
helper.registerAllowedMethod("httpRequest", [Map], {Map parameters ->
|
|
|
|
switch (parameters.url) {
|
|
|
|
case customDefaultUrl:
|
|
|
|
urlRequested = true
|
|
|
|
return [status: 200, content: "custom: 'myRemoteConfig'"]
|
|
|
|
default:
|
|
|
|
throw new IllegalArgumentException('wrong URL requested')
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
helper.registerAllowedMethod("readYaml", [Map], { Map parameters ->
|
|
|
|
Yaml yamlParser = new Yaml()
|
|
|
|
if (parameters.text) {
|
|
|
|
return yamlParser.load(parameters.text)
|
|
|
|
} else if (parameters.file) {
|
|
|
|
if (parameters.file == '.pipeline/config-with-custom-defaults.yml') {
|
|
|
|
return [customDefaults: "${customDefaultUrl}"]
|
|
|
|
}
|
|
|
|
if (parameters.file == '.pipeline/custom_default_from_url_0.yml') {
|
|
|
|
return [custom: 'myRemoteConfig']
|
|
|
|
}
|
|
|
|
}
|
|
|
|
throw new IllegalArgumentException("Unexpected invocation of readYaml step")
|
|
|
|
})
|
|
|
|
|
|
|
|
stepRule.step.setupCommonPipelineEnvironment(
|
|
|
|
script: nullScript,
|
|
|
|
customDefaults: 'custom.yml',
|
|
|
|
configFile: '.pipeline/config-with-custom-defaults.yml',
|
|
|
|
)
|
|
|
|
assertEquals("custom: 'myRemoteConfig'", writeFileRule.files['.pipeline/custom_default_from_url_0.yml'])
|
|
|
|
assertEquals('myRemoteConfig', DefaultValueCache.instance.defaultValues['custom'])
|
|
|
|
}
|
2020-09-04 14:45:09 +02:00
|
|
|
|
|
|
|
|
|
|
|
@Test
|
|
|
|
void inferBuildToolMaven() {
|
|
|
|
helper.registerAllowedMethod('fileExists', [String.class], { s ->
|
|
|
|
return s == "pom.xml"
|
|
|
|
})
|
|
|
|
setupCommonPipelineEnvironment.inferBuildTool(nullScript, [inferBuildTool: true])
|
|
|
|
assertEquals('maven', nullScript.commonPipelineEnvironment.buildTool)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
void inferBuildToolMTA() {
|
|
|
|
helper.registerAllowedMethod('fileExists', [String.class], { s ->
|
|
|
|
return s == "mta.yaml"
|
|
|
|
})
|
|
|
|
setupCommonPipelineEnvironment.inferBuildTool(nullScript, [inferBuildTool: true])
|
|
|
|
assertEquals('mta', nullScript.commonPipelineEnvironment.buildTool)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
void inferBuildToolNpm() {
|
|
|
|
helper.registerAllowedMethod('fileExists', [String.class], { s ->
|
|
|
|
return s == "package.json"
|
|
|
|
})
|
|
|
|
setupCommonPipelineEnvironment.inferBuildTool(nullScript, [inferBuildTool: true])
|
|
|
|
assertEquals('npm', nullScript.commonPipelineEnvironment.buildTool)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
void inferBuildToolNone() {
|
|
|
|
helper.registerAllowedMethod('fileExists', [String.class], { s ->
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
setupCommonPipelineEnvironment.inferBuildTool(nullScript, [inferBuildTool: true])
|
|
|
|
assertNull(nullScript.commonPipelineEnvironment.buildTool)
|
|
|
|
}
|
|
|
|
|
2020-10-05 12:50:03 +02:00
|
|
|
@Test
|
2020-11-11 11:44:49 +02:00
|
|
|
void "Set scmInfo parameter sets commit id"() {
|
2020-10-05 12:50:03 +02:00
|
|
|
helper.registerAllowedMethod("fileExists", [String], { String path ->
|
|
|
|
return path.endsWith('.pipeline/config.yml')
|
|
|
|
})
|
|
|
|
|
2020-11-11 11:44:49 +02:00
|
|
|
def dummyScmInfo = [GIT_COMMIT: 'dummy_git_commit_id', GIT_URL: 'https://github.com/testOrg/testRepo.git']
|
|
|
|
|
|
|
|
stepRule.step.setupCommonPipelineEnvironment(script: nullScript, scmInfo: dummyScmInfo)
|
|
|
|
assertThat(nullScript.commonPipelineEnvironment.gitCommitId, is('dummy_git_commit_id'))
|
2020-10-05 12:50:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
2020-11-11 11:44:49 +02:00
|
|
|
void "No scmInfo passed as parameter yields empty git info"() {
|
2020-10-05 12:50:03 +02:00
|
|
|
helper.registerAllowedMethod("fileExists", [String], { String path ->
|
|
|
|
return path.endsWith('.pipeline/config.yml')
|
|
|
|
})
|
|
|
|
|
2020-11-11 11:44:49 +02:00
|
|
|
stepRule.step.setupCommonPipelineEnvironment(script: nullScript)
|
|
|
|
assertNull(nullScript.commonPipelineEnvironment.gitCommitId)
|
2020-10-05 12:50:03 +02:00
|
|
|
assertNull(nullScript.commonPipelineEnvironment.getGitSshUrl())
|
|
|
|
assertNull(nullScript.commonPipelineEnvironment.getGitHttpsUrl())
|
|
|
|
assertNull(nullScript.commonPipelineEnvironment.getGithubOrg())
|
|
|
|
assertNull(nullScript.commonPipelineEnvironment.getGithubRepo())
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
void testSetScmInfoOnCommonPipelineEnvironment() {
|
|
|
|
//currently supported formats
|
|
|
|
def scmInfoTestList = [
|
|
|
|
[GIT_URL: 'https://github.com/testOrg/testRepo.git', expectedSsh: 'git@github.com:testOrg/testRepo.git', expectedHttp: 'https://github.com/testOrg/testRepo.git', expectedOrg: 'testOrg', expectedRepo: 'testRepo'],
|
|
|
|
[GIT_URL: 'https://github.com:7777/testOrg/testRepo.git', expectedSsh: 'git@github.com:testOrg/testRepo.git', expectedHttp: 'https://github.com:7777/testOrg/testRepo.git', expectedOrg: 'testOrg', expectedRepo: 'testRepo'],
|
|
|
|
[GIT_URL: 'git@github.com:testOrg/testRepo.git', expectedSsh: 'git@github.com:testOrg/testRepo.git', expectedHttp: 'https://github.com/testOrg/testRepo.git', expectedOrg: 'testOrg', expectedRepo: 'testRepo'],
|
|
|
|
[GIT_URL: 'ssh://git@github.com/testOrg/testRepo.git', expectedSsh: 'ssh://git@github.com/testOrg/testRepo.git', expectedHttp: 'https://github.com/testOrg/testRepo.git', expectedOrg: 'testOrg', expectedRepo: 'testRepo'],
|
|
|
|
[GIT_URL: 'ssh://git@github.com:7777/testOrg/testRepo.git', expectedSsh: 'ssh://git@github.com:7777/testOrg/testRepo.git', expectedHttp: 'https://github.com/testOrg/testRepo.git', expectedOrg: 'testOrg', expectedRepo: 'testRepo'],
|
|
|
|
[GIT_URL: 'ssh://git@github.com/path/to/testOrg/testRepo.git', expectedSsh: 'ssh://git@github.com/path/to/testOrg/testRepo.git', expectedHttp: 'https://github.com/path/to/testOrg/testRepo.git', expectedOrg: 'path/to/testOrg', expectedRepo: 'testRepo'],
|
2020-11-03 13:50:00 +02:00
|
|
|
[GIT_URL: 'ssh://git@github.com/path/testOrg/testRepo.git', expectedSsh: 'ssh://git@github.com/path/testOrg/testRepo.git', expectedHttp: 'https://github.com/path/testOrg/testRepo.git', expectedOrg: 'path/testOrg', expectedRepo: 'testRepo'],
|
2020-10-05 12:50:03 +02:00
|
|
|
[GIT_URL: 'ssh://git@github.com/testRepo.git', expectedSsh: 'ssh://git@github.com/testRepo.git', expectedHttp: 'https://github.com/testRepo.git', expectedOrg: 'N/A', expectedRepo: 'testRepo'],
|
|
|
|
]
|
|
|
|
|
|
|
|
scmInfoTestList.each {scmInfoTest ->
|
|
|
|
stepRule.step.setupCommonPipelineEnvironment.setGitUrlsOnCommonPipelineEnvironment(nullScript, scmInfoTest.GIT_URL)
|
|
|
|
assertThat(nullScript.commonPipelineEnvironment.getGitSshUrl(), is(scmInfoTest.expectedSsh))
|
|
|
|
assertThat(nullScript.commonPipelineEnvironment.getGitHttpsUrl(), is(scmInfoTest.expectedHttp))
|
|
|
|
assertThat(nullScript.commonPipelineEnvironment.getGithubOrg(), is(scmInfoTest.expectedOrg))
|
|
|
|
assertThat(nullScript.commonPipelineEnvironment.getGithubRepo(), is(scmInfoTest.expectedRepo))
|
|
|
|
}
|
|
|
|
}
|
2017-12-06 13:03:06 +02:00
|
|
|
}
|
2019-03-26 15:13:03 +02:00
|
|
|
|