Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
C
CONF
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
UvA
CONF
Commits
09b05117
Commit
09b05117
authored
Feb 14, 2017
by
zh9314
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add a consumer to invoke provisioner.
parent
5a5e0fe5
Changes
5
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
469 additions
and
0 deletions
+469
-0
nb-configuration.xml
drip-provisioner/nb-configuration.xml
+19
-0
pom.xml
drip-provisioner/pom.xml
+41
-0
.DS_Store
.../src/main/java/nl/uva/sne/drip/drip/provisioner/.DS_Store
+0
-0
Consumer.java
.../main/java/nl/uva/sne/drip/drip/provisioner/Consumer.java
+340
-0
RPCServer.java
...main/java/nl/uva/sne/drip/drip/provisioner/RPCServer.java
+69
-0
No files found.
drip-provisioner/nb-configuration.xml
0 → 100644
View file @
09b05117
<?xml version="1.0" encoding="UTF-8"?>
<project-shared-configuration>
<!--
This file contains additional configuration written by modules in the NetBeans IDE.
The configuration is intended to be shared among all the users of project and
therefore it is assumed to be part of version control checkout.
Without this configuration present, some functionality in the IDE may be limited or fail altogether.
-->
<properties
xmlns=
"http://www.netbeans.org/ns/maven-properties-data/1"
>
<!--
Properties that influence various parts of the IDE, especially code formatting and the like.
You can copy and paste the single properties, into the pom.xml file and the IDE will pick them up.
That way multiple projects can share the same settings (useful for formatting rules for example).
Any value defined here will override the pom.xml file value but is only applicable to the current project.
-->
<netbeans.hint.licensePath>
${project.basedir}/../licenseheader.txt
</netbeans.hint.licensePath>
<org-netbeans-modules-whitelist.whitelist-oracle>
false
</org-netbeans-modules-whitelist.whitelist-oracle>
</properties>
</project-shared-configuration>
drip-provisioner/pom.xml
0 → 100644
View file @
09b05117
<?xml version="1.0" encoding="UTF-8"?>
<project
xmlns=
"http://maven.apache.org/POM/4.0.0"
xmlns:xsi=
"http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation=
"http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"
>
<modelVersion>
4.0.0
</modelVersion>
<parent>
<groupId>
nl.uva.sne.drip
</groupId>
<artifactId>
drip
</artifactId>
<version>
1.0-SNAPSHOT
</version>
</parent>
<artifactId>
drip-simple_provisioner
</artifactId>
<packaging>
jar
</packaging>
<properties>
<maven.compiler.source>
1.8
</maven.compiler.source>
<maven.compiler.target>
1.8
</maven.compiler.target>
</properties>
<dependencies>
<dependency>
<groupId>
nl.uva.sne.drip
</groupId>
<artifactId>
drip-commons
</artifactId>
<version>
1.0-SNAPSHOT
</version>
</dependency>
<dependency>
<groupId>
com.rabbitmq
</groupId>
<artifactId>
amqp-client
</artifactId>
<version>
4.0.2
</version>
</dependency>
<dependency>
<groupId>
com.fasterxml.jackson.core
</groupId>
<artifactId>
jackson-databind
</artifactId>
<version>
2.8.6
</version>
</dependency>
<dependency>
<groupId>
org.json
</groupId>
<artifactId>
json
</artifactId>
<version>
20090211
</version>
<type>
jar
</type>
</dependency>
</dependencies>
</project>
\ No newline at end of file
drip-provisioner/src/main/java/nl/uva/sne/drip/drip/provisioner/.DS_Store
0 → 100644
View file @
09b05117
File added
drip-provisioner/src/main/java/nl/uva/sne/drip/drip/provisioner/Consumer.java
0 → 100644
View file @
09b05117
/*
* Copyright 2017 S. Koulouzis, Wang Junchao, Huan Zhou, Yang Hu
*
* 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
nl
.
uva
.
sne
.
drip
.
drip
.
planner
;
import
com.fasterxml.jackson.databind.ObjectMapper
;
import
com.rabbitmq.client.AMQP
;
import
com.rabbitmq.client.Channel
;
import
com.rabbitmq.client.DefaultConsumer
;
import
com.rabbitmq.client.Envelope
;
import
java.io.File
;
import
java.io.FileNotFoundException
;
import
java.io.IOException
;
import
java.io.PrintWriter
;
import
java.io.UnsupportedEncodingException
;
import
java.io.InputStreamReader
;
import
java.nio.file.Files
;
import
java.nio.file.Paths
;
import
java.util.ArrayList
;
import
java.util.HashMap
;
import
java.util.List
;
import
java.util.Map
;
import
java.util.logging.Level
;
import
java.util.logging.Logger
;
import
nl.uva.sne.drip.commons.types.Parameter
;
import
nl.uva.sne.drip.commons.types.Message
;
import
org.json.JSONArray
;
import
org.json.JSONException
;
import
org.json.JSONObject
;
/**
*
* This is an example of a Message consumer
*
*
* @author H. Zhou
*/
public
class
Consumer
extends
DefaultConsumer
{
private
final
Channel
channel
;
private
final
Planner
panner
;
private
final
String
jarFilePath
=
"/root/SWITCH/bin/ProvisioningCore.jar"
;
public
class
topologyElement
{
String
topologyName
=
""
;
String
outputFilePath
=
""
;
}
public
Consumer
(
Channel
channel
)
{
super
(
channel
);
this
.
channel
=
channel
;
this
.
panner
=
new
Planner
();
}
@Override
public
void
handleDelivery
(
String
consumerTag
,
Envelope
envelope
,
AMQP
.
BasicProperties
properties
,
byte
[]
body
)
throws
IOException
{
//Create the reply properties which tells us where to reply, and which id to use.
//No need to change anything here
AMQP
.
BasicProperties
replyProps
=
new
AMQP
.
BasicProperties
.
Builder
()
.
correlationId
(
properties
.
getCorrelationId
())
.
build
();
String
response
=
""
;
try
{
//The queue only moves bytes so we need to convert them to stting
String
message
=
new
String
(
body
,
"UTF-8"
);
//Create a tmp folder to save the output.
String
tempInputDirPath
=
System
.
getProperty
(
"java.io.tmpdir"
)
+
File
.
separator
+
"Input-"
+
Long
.
toString
(
System
.
nanoTime
())
+
File
.
separator
;
File
tempInputDir
=
new
File
(
tempInputDirPath
);
if
(!(
tempInputDir
.
mkdirs
()))
{
throw
new
FileNotFoundException
(
"Could not create input directory: "
+
tempInputDir
.
getAbsolutePath
());
}
ArrayList
topologyInfoArray
=
null
;
topologyInfoArray
=
invokeProvisioner
(
message
,
tempInputDirPath
);
response
=
generateResponse
(
topologyInfoArray
);
}
catch
(
IOException
|
JSONException
ex
)
{
response
=
ex
.
getMessage
();
Logger
.
getLogger
(
Consumer
.
class
.
getName
()).
log
(
Level
.
SEVERE
,
null
,
ex
);
}
finally
{
//We send the response back. No need to change anything here
channel
.
basicPublish
(
""
,
properties
.
getReplyTo
(),
replyProps
,
response
.
getBytes
(
"UTF-8"
));
channel
.
basicAck
(
envelope
.
getDeliveryTag
(),
false
);
}
}
////If the provisioner jar file is successfully invoked, the returned value should be a set of output file paths which are expected.
////If there are some errors or some information missing with this message, the returned value will be null.
////The input dir path contains '/'
private
ArrayList
<
topologyElement
>
invokeProvisioner
(
String
message
,
String
tempInputDirPath
)
throws
IOException
{
//Use the Jackson API to convert json to Object
JSONObject
jo
=
new
JSONObject
(
message
);
JSONArray
parameters
=
jo
.
getJSONArray
(
"parameters"
);
//Create tmp input files
File
ec2ConfFile
=
null
;
File
geniConfFile
=
null
;
//loop through the parameters in a message to find the input files
String
logDir
=
"null"
,
mainTopologyPath
=
"null"
,
sshKeyFilePath
=
"null"
,
scriptPath
=
"null"
;
ArrayList
<
topologyElement
>
topologyInfoArray
=
new
ArrayList
<
topologyElement
>();
for
(
int
i
=
0
;
i
<
parameters
.
length
();
i
++)
{
JSONObject
param
=
(
JSONObject
)
parameters
.
get
(
i
);
String
name
=
(
String
)
param
.
get
(
Parameter
.
NAME
);
if
(
name
.
equals
(
"ec2.conf"
))
{
try
{
ec2ConfFile
=
new
File
(
tempInputDirPath
+
"ec2.Conf"
);
if
(
ec2ConfFile
.
createNewFile
()){
PrintWriter
out
=
new
PrintWriter
(
ec2ConfFile
);
out
.
print
(
param
.
get
(
Parameter
.
VALUE
));
}
else
return
null
;
}
catch
(
IOException
e
)
{
e
.
printStackTrace
();
return
null
;
}
}
else
if
(
name
.
equals
(
"geni.conf"
)){
try
{
geniConfFile
=
new
File
(
tempInputDirPath
+
"geni.Conf"
);
if
(
geniConfFile
.
createNewFile
()){
PrintWriter
out
=
new
PrintWriter
(
geniConfFile
);
out
.
print
(
param
.
get
(
Parameter
.
VALUE
));
}
else
return
null
;
}
catch
(
IOException
e
)
{
e
.
printStackTrace
();
return
null
;
}
}
else
if
(
name
.
equals
(
"topology"
)){
JSONObject
attribute_level
=
param
.
getJSONObject
(
"attributes"
);
int
fileLevel
=
Integer
.
valueOf
(
attribute_level
.
get
(
"level"
));
if
(
fileLevel
==
0
)
/////if the file level is 0, it means that this is the top level description
{
try
{
File
topologyFile
=
new
File
(
tempInputDirPath
+
"topology_main"
);
if
(
topologyFile
.
createNewFile
()){
PrintWriter
out
=
new
PrintWriter
(
geniConfFile
);
out
.
print
(
param
.
get
(
Parameter
.
VALUE
));
mainTopologyPath
=
topologyFile
.
getAbsolutePath
();
}
else
return
null
;
}
catch
(
IOException
e
)
{
e
.
printStackTrace
();
return
null
;
}
}
else
if
(
fileLevel
==
1
){
////this means that this file is the low level detailed description
String
fileName
=
attribute_level
.
get
(
"filename"
);
////This file name does not contain suffix of '.yml' for example
try
{
File
topologyFile
=
new
File
(
tempInputDirPath
+
fileName
+
".yml"
);
String
outputFilePath
=
tempInputDirPath
+
fileName
+
"_provisioned.yml"
;
if
(
topologyFile
.
createNewFile
()){
PrintWriter
out
=
new
PrintWriter
(
geniConfFile
);
out
.
print
(
param
.
get
(
Parameter
.
VALUE
));
topologyElement
x
=
new
topologyElement
();
x
.
topologyName
=
fileName
;
x
.
outputFilePath
=
outputFilePath
;
topologyInfoArray
.
add
(
x
);
}
else
return
null
;
}
catch
(
IOException
e
)
{
e
.
printStackTrace
();
return
null
;
}
}
}
else
if
(
name
.
equals
(
"logdir"
)){
logDir
=
param
.
get
(
Parameter
.
VALUE
);
}
else
if
(
name
.
equals
(
"sshkey"
)){
sshKeyFilePath
=
param
.
get
(
Parameter
.
VALUE
);
}
else
if
(
name
.
equals
(
"guiscript"
)){
scriptPath
=
param
.
get
(
Parameter
.
VALUE
);
}
else
{
return
null
;
}
}
File
curDir
=
new
File
(
tempInputDirPath
);
String
[]
ls
=
curDir
.
list
();
for
(
int
i
=
0
;
i
<
ls
.
length
;
i
++){
if
(
ls
[
i
].
contains
(
"."
)){
String
[]
fileTypes
=
ls
[
i
].
split
(
"\\."
);
if
(
fileTypes
.
length
>
0
){
int
lastIndex
=
fileTypes
.
length
-
1
;
String
fileType
=
fileTypes
[
lastIndex
];
if
(
fileType
.
equals
(
"yml"
)){
String
toscaFile
=
curDir
+
ls
[
i
];
if
(!
sshKeyFilePath
.
equals
(
"null"
))
changeKeyFilePath
(
toscaFile
,
sshKeyFilePath
);
if
(!
scriptPath
.
equals
(
"null"
))
changeGUIScriptFilePath
(
toscaFile
,
scriptPath
);
}
}
}
}
if
(
ec2ConfFile
==
null
&&
geniConfFile
==
null
)
return
null
;
if
(
mainTopologyPath
.
equals
(
"null"
))
return
null
;
String
ec2ConfFilePath
=
"null"
;
String
geniConfFilePath
=
"null"
;
if
(
ec2ConfFile
!=
null
)
ec2ConfFilePath
=
ec2ConfFile
.
getAbsolutePath
();
if
(
geniConfFile
!=
null
)
geniConfFilePath
=
geniConfFile
.
getAbsolutePath
();
if
(
logDir
.
equals
(
"null"
))
logDir
=
"/tmp/"
;
String
cmd
=
"java -jar "
+
jarFilePath
+
" ec2="
+
ec2ConfFilePath
+
" exogeni="
+
geniConfFilePath
+
" logDir="
+
logDir
+
" topology="
+
mainTopologyPath
;
try
{
Process
p
=
Runtime
.
getRuntime
().
exec
(
cmd
);
p
.
waitFor
();
}
catch
(
IOException
|
InterruptedException
e
)
{
// TODO Auto-generated catch block
e
.
printStackTrace
();
}
return
topologyInfoArray
;
}
////Change the key file path in the tosca file.
////Because the user needs to upload their public key file into the server file system.
private
void
changeKeyFilePath
(
String
toscaFilePath
,
String
newKeyFilePath
){
File
toscaFile
=
new
File
(
toscaFilePath
);
String
fileContent
=
""
;
try
{
BufferedReader
in
=
new
BufferedReader
(
new
FileReader
(
toscaFile
));
String
line
=
""
;
while
((
line
=
in
.
readLine
())
!=
null
){
if
(
line
.
contains
(
"publicKeyPath"
))
fileContent
+=
(
"publicKeyPath: "
+
newKeyFilePath
+
"\n"
);
else
fileContent
+=
(
line
+
"\n"
);
}
in
.
close
();
FileWriter
fw
=
new
FileWriter
(
toscaFilePath
,
false
);
fw
.
write
(
fileContent
);
fw
.
close
();
}
catch
(
FileNotFoundException
e
)
{
// TODO Auto-generated catch block
e
.
printStackTrace
();
}
catch
(
IOException
e
)
{
// TODO Auto-generated catch block
e
.
printStackTrace
();
}
}
private
void
changeGUIScriptFilePath
(
String
toscaFilePath
,
String
newScriptPath
){
File
toscaFile
=
new
File
(
toscaFilePath
);
String
fileContent
=
""
;
try
{
BufferedReader
in
=
new
BufferedReader
(
new
FileReader
(
toscaFile
));
String
line
=
""
;
while
((
line
=
in
.
readLine
())
!=
null
){
if
(
line
.
contains
(
"script"
)){
int
index
=
line
.
indexOf
(
"script:"
);
String
prefix
=
line
.
substring
(
0
,
index
+
7
);
fileContent
+=
(
prefix
+
" "
+
newScriptPath
+
"\n"
);
}
else
fileContent
+=
(
line
+
"\n"
);
}
in
.
close
();
FileWriter
fw
=
new
FileWriter
(
toscaFilePath
,
false
);
fw
.
write
(
fileContent
);
fw
.
close
();
}
catch
(
FileNotFoundException
e
)
{
// TODO Auto-generated catch block
e
.
printStackTrace
();
}
catch
(
IOException
e
)
{
// TODO Auto-generated catch block
e
.
printStackTrace
();
}
}
private
String
generateResponse
(
ArrayList
<
topologyElement
>
outputs
)
throws
JSONException
,
IOException
{
//Use the JSONObject API to convert Object (Message) to json
JSONObject
jo
=
new
JSONObject
();
jo
.
put
(
"creationDate"
,
(
System
.
currentTimeMillis
()));
List
parameters
=
new
ArrayList
();
String
charset
=
"UTF-8"
;
if
(
outputs
==
null
){
Map
<
String
,
String
>
fileArguments
=
new
HashMap
<>();
fileArguments
.
put
(
"encoding"
,
charset
);
fileArguments
.
put
(
"name"
,
"ERROR"
);
fileArguments
.
put
(
"value"
,
"Some error with input messages!"
);
parameters
.
add
(
fileArguments
);
}
else
{
for
(
int
i
=
0
;
i
<
outputs
.
size
()
;
i
++){
Map
<
String
,
String
>
fileArguments
=
new
HashMap
<>();
fileArguments
.
put
(
"encoding"
,
charset
);
File
f
=
new
File
(
outputs
.
get
(
i
).
outputFilePath
);
if
(
f
.
exists
()){
fileArguments
.
put
(
"name"
,
outputs
.
get
(
i
).
topologyName
);
byte
[]
bytes
=
Files
.
readAllBytes
(
Paths
.
get
(
f
.
getAbsolutePath
()));
fileArguments
.
put
(
"value"
,
new
String
(
bytes
,
charset
));
parameters
.
add
(
fileArguments
);
}
else
{
fileArguments
.
put
(
"name"
,
outputs
.
get
(
i
).
topologyName
);
fileArguments
.
put
(
"value"
,
"ERROR::There is no output for topology "
+
outputs
.
get
(
i
).
topologyName
);
parameters
.
add
(
fileArguments
);
}
}
}
jo
.
put
(
"parameters"
,
parameters
);
return
jo
.
toString
();
}
}
drip-provisioner/src/main/java/nl/uva/sne/drip/drip/provisioner/RPCServer.java
0 → 100644
View file @
09b05117
/*
* Copyright 2017 S. Koulouzis, Wang Junchao, Huan Zhou, Yang Hu
*
* 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
nl
.
uva
.
sne
.
drip
.
drip
.
provisioner
;
import
com.rabbitmq.client.AMQP
;
import
com.rabbitmq.client.Channel
;
import
com.rabbitmq.client.Connection
;
import
com.rabbitmq.client.ConnectionFactory
;
import
java.io.IOException
;
import
java.util.concurrent.TimeoutException
;
import
java.util.logging.Level
;
import
java.util.logging.Logger
;
/**
*
* @author H. Zhou
*/
public
class
RPCServer
{
private
static
final
String
RPC_QUEUE_NAME
=
"provisioner_queue"
;
private
static
final
String
HOST
=
"172.17.0.3"
;
public
static
void
main
(
String
[]
argv
)
{
start
();
}
private
static
void
start
()
{
ConnectionFactory
factory
=
new
ConnectionFactory
();
factory
.
setHost
(
HOST
);
factory
.
setPassword
(
"guest"
);
factory
.
setUsername
(
"guest"
);
factory
.
setPort
(
AMQP
.
PROTOCOL
.
PORT
);
try
(
Connection
connection
=
factory
.
newConnection
())
{
Channel
channel
=
connection
.
createChannel
();
//We define the queue name
channel
.
queueDeclare
(
RPC_QUEUE_NAME
,
false
,
false
,
false
,
null
);
//Set our own customized consummer
Consumer
c
=
new
Consumer
(
channel
);
//Start listening for messages
channel
.
basicConsume
(
RPC_QUEUE_NAME
,
false
,
c
);
//Block so we don't close the channel
while
(
true
)
{
try
{
Thread
.
sleep
(
100
);
}
catch
(
InterruptedException
_ignore
)
{
}
}
}
catch
(
IOException
|
TimeoutException
ex
)
{
Logger
.
getLogger
(
RPCServer
.
class
.
getName
()).
log
(
Level
.
SEVERE
,
null
,
ex
);
}
}
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment