Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
O
OpenXG-RAN
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
lizhongxiao
OpenXG-RAN
Commits
41a9ac11
Commit
41a9ac11
authored
Jul 20, 2022
by
Robert Schmidt
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Python Class for Cluster Building and Deployment
parent
ebdbb1fc
Changes
8
Hide whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
484 additions
and
11 deletions
+484
-11
ci-scripts/args_parse.py
ci-scripts/args_parse.py
+19
-1
ci-scripts/cls_cluster.py
ci-scripts/cls_cluster.py
+407
-0
ci-scripts/main.py
ci-scripts/main.py
+9
-2
ci-scripts/xml_class_list.yml
ci-scripts/xml_class_list.yml
+1
-0
ci-scripts/xml_files/cluster_image_build.xml
ci-scripts/xml_files/cluster_image_build.xml
+35
-0
openshift/oai-physim-bc.yaml
openshift/oai-physim-bc.yaml
+0
-4
openshift/ran-base-log-retrieval.yaml
openshift/ran-base-log-retrieval.yaml
+13
-0
openshift/ran-build-bc.yaml
openshift/ran-build-bc.yaml
+0
-4
No files found.
ci-scripts/args_parse.py
View file @
41a9ac11
...
...
@@ -41,7 +41,7 @@ import constants as CONST
#-----------------------------------------------------------
def
ArgsParse
(
argvs
,
CiTestObj
,
RAN
,
HTML
,
EPC
,
ldpc
,
CONTAINERS
,
HELP
,
SCA
,
PHYSIM
):
def
ArgsParse
(
argvs
,
CiTestObj
,
RAN
,
HTML
,
EPC
,
ldpc
,
CONTAINERS
,
HELP
,
SCA
,
PHYSIM
,
CLUSTER
):
py_param_file_present
=
False
...
...
@@ -82,6 +82,7 @@ def ArgsParse(argvs,CiTestObj,RAN,HTML,EPC,ldpc,CONTAINERS,HELP,SCA,PHYSIM):
CONTAINERS
.
ranRepository
=
matchReg
.
group
(
1
)
SCA
.
ranRepository
=
matchReg
.
group
(
1
)
PHYSIM
.
ranRepository
=
matchReg
.
group
(
1
)
CLUSTER
.
ranRepository
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-eNB_AllowMerge=(.+)$|^\-\-ranAllowMerge=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
if
re
.
match
(
'^\-\-eNB_AllowMerge=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-eNB_AllowMerge=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
...
...
@@ -96,6 +97,7 @@ def ArgsParse(argvs,CiTestObj,RAN,HTML,EPC,ldpc,CONTAINERS,HELP,SCA,PHYSIM):
CONTAINERS
.
ranAllowMerge
=
True
SCA
.
ranAllowMerge
=
True
PHYSIM
.
ranAllowMerge
=
True
CLUSTER
.
ranAllowMerge
=
True
elif
re
.
match
(
'^\-\-eNBBranch=(.+)$|^\-\-ranBranch=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
if
re
.
match
(
'^\-\-eNBBranch=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-eNBBranch=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
...
...
@@ -108,6 +110,7 @@ def ArgsParse(argvs,CiTestObj,RAN,HTML,EPC,ldpc,CONTAINERS,HELP,SCA,PHYSIM):
CONTAINERS
.
ranBranch
=
matchReg
.
group
(
1
)
SCA
.
ranBranch
=
matchReg
.
group
(
1
)
PHYSIM
.
ranBranch
=
matchReg
.
group
(
1
)
CLUSTER
.
ranBranch
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-eNBCommitID=(.*)$|^\-\-ranCommitID=(.*)$'
,
myArgv
,
re
.
IGNORECASE
):
if
re
.
match
(
'^\-\-eNBCommitID=(.*)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-eNBCommitID=(.*)$'
,
myArgv
,
re
.
IGNORECASE
)
...
...
@@ -120,6 +123,7 @@ def ArgsParse(argvs,CiTestObj,RAN,HTML,EPC,ldpc,CONTAINERS,HELP,SCA,PHYSIM):
CONTAINERS
.
ranCommitID
=
matchReg
.
group
(
1
)
SCA
.
ranCommitID
=
matchReg
.
group
(
1
)
PHYSIM
.
ranCommitID
=
matchReg
.
group
(
1
)
CLUSTER
.
ranCommitID
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-eNBTargetBranch=(.*)$|^\-\-ranTargetBranch=(.*)$'
,
myArgv
,
re
.
IGNORECASE
):
if
re
.
match
(
'^\-\-eNBTargetBranch=(.*)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-eNBTargetBranch=(.*)$'
,
myArgv
,
re
.
IGNORECASE
)
...
...
@@ -132,6 +136,7 @@ def ArgsParse(argvs,CiTestObj,RAN,HTML,EPC,ldpc,CONTAINERS,HELP,SCA,PHYSIM):
CONTAINERS
.
ranTargetBranch
=
matchReg
.
group
(
1
)
SCA
.
ranTargetBranch
=
matchReg
.
group
(
1
)
PHYSIM
.
ranTargetBranch
=
matchReg
.
group
(
1
)
CLUSTER
.
ranTargetBranch
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-eNBIPAddress=(.+)$|^\-\-eNB[1-2]IPAddress=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
if
re
.
match
(
'^\-\-eNBIPAddress=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-eNBIPAddress=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
...
...
@@ -140,6 +145,7 @@ def ArgsParse(argvs,CiTestObj,RAN,HTML,EPC,ldpc,CONTAINERS,HELP,SCA,PHYSIM):
CONTAINERS
.
eNBIPAddress
=
matchReg
.
group
(
1
)
SCA
.
eNBIPAddress
=
matchReg
.
group
(
1
)
PHYSIM
.
eNBIPAddress
=
matchReg
.
group
(
1
)
CLUSTER
.
eNBIPAddress
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-eNB1IPAddress=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-eNB1IPAddress=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
RAN
.
eNB1IPAddress
=
matchReg
.
group
(
1
)
...
...
@@ -156,6 +162,7 @@ def ArgsParse(argvs,CiTestObj,RAN,HTML,EPC,ldpc,CONTAINERS,HELP,SCA,PHYSIM):
CONTAINERS
.
eNBUserName
=
matchReg
.
group
(
1
)
SCA
.
eNBUserName
=
matchReg
.
group
(
1
)
PHYSIM
.
eNBUserName
=
matchReg
.
group
(
1
)
CLUSTER
.
eNBUserName
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-eNB1UserName=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-eNB1UserName=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
RAN
.
eNB1UserName
=
matchReg
.
group
(
1
)
...
...
@@ -172,6 +179,7 @@ def ArgsParse(argvs,CiTestObj,RAN,HTML,EPC,ldpc,CONTAINERS,HELP,SCA,PHYSIM):
CONTAINERS
.
eNBPassword
=
matchReg
.
group
(
1
)
SCA
.
eNBPassword
=
matchReg
.
group
(
1
)
PHYSIM
.
eNBPassword
=
matchReg
.
group
(
1
)
CLUSTER
.
eNBPassword
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-eNB1Password=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-eNB1Password=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
RAN
.
eNB1Password
=
matchReg
.
group
(
1
)
...
...
@@ -188,6 +196,7 @@ def ArgsParse(argvs,CiTestObj,RAN,HTML,EPC,ldpc,CONTAINERS,HELP,SCA,PHYSIM):
CONTAINERS
.
eNBSourceCodePath
=
matchReg
.
group
(
1
)
SCA
.
eNBSourceCodePath
=
matchReg
.
group
(
1
)
PHYSIM
.
eNBSourceCodePath
=
matchReg
.
group
(
1
)
CLUSTER
.
eNBSourceCodePath
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-eNB1SourceCodePath=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-eNB1SourceCodePath=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
RAN
.
eNB1SourceCodePath
=
matchReg
.
group
(
1
)
...
...
@@ -260,12 +269,21 @@ def ArgsParse(argvs,CiTestObj,RAN,HTML,EPC,ldpc,CONTAINERS,HELP,SCA,PHYSIM):
elif
re
.
match
(
'^\-\-OCUserName=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-OCUserName=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
PHYSIM
.
OCUserName
=
matchReg
.
group
(
1
)
CLUSTER
.
OCUserName
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-OCPassword=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-OCPassword=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
PHYSIM
.
OCPassword
=
matchReg
.
group
(
1
)
CLUSTER
.
OCPassword
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-OCProjectName=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-OCProjectName=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
PHYSIM
.
OCProjectName
=
matchReg
.
group
(
1
)
CLUSTER
.
OCProjectName
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-OCUrl=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-OCUrl=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
CLUSTER
.
OCUrl
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-OCRegistry=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-OCRegistry=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
CLUSTER
.
OCRegistry
=
matchReg
.
group
(
1
)
elif
re
.
match
(
'^\-\-BuildId=(.+)$'
,
myArgv
,
re
.
IGNORECASE
):
matchReg
=
re
.
match
(
'^\-\-BuildId=(.+)$'
,
myArgv
,
re
.
IGNORECASE
)
RAN
.
BuildId
=
matchReg
.
group
(
1
)
...
...
ci-scripts/cls_cluster.py
0 → 100644
View file @
41a9ac11
#/*
# * Licensed to the OpenAirInterface (OAI) Software Alliance under one or more
# * contributor license agreements. See the NOTICE file distributed with
# * this work for additional information regarding copyright ownership.
# * The OpenAirInterface Software Alliance licenses this file to You under
# * the OAI Public License, Version 1.1 (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.openairinterface.org/?page_id=698
# *
# * 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.
# *-------------------------------------------------------------------------------
# * For more information about the OpenAirInterface (OAI) Software Alliance:
# * contact@openairinterface.org
# */
#---------------------------------------------------------------------
# Python for CI of OAI-eNB + COTS-UE
#
# Required Python Version
# Python 3.x
#
# Required Python Package
# pexpect
#---------------------------------------------------------------------
#-----------------------------------------------------------
# Import
#-----------------------------------------------------------
import
logging
import
sshconnection
as
SSH
import
cls_oai_html
#import os
import
re
import
time
#import subprocess
import
sys
import
constants
as
CONST
import
helpreadme
as
HELP
import
cls_containerize
class
Cluster
:
def
__init__
(
self
):
self
.
eNBIPAddress
=
""
self
.
eNBSourceCodePath
=
""
self
.
forcedWorkspaceCleanup
=
False
self
.
OCUserName
=
""
self
.
OCPassword
=
""
self
.
OCProjectName
=
""
self
.
OCUrl
=
"https://api.oai.cs.eurecom.fr:6443"
self
.
OCRegistry
=
"default-route-openshift-image-registry.apps.oai.cs.eurecom.fr/"
self
.
ranRepository
=
""
self
.
ranBranch
=
""
self
.
ranCommitID
=
""
self
.
ranAllowMerge
=
False
self
.
ranTargetBranch
=
""
def
_createWorkspace
(
self
,
sshSession
,
password
,
sourcePath
):
# on RedHat/CentOS .git extension is mandatory
result
=
re
.
search
(
'([a-zA-Z0-9\:\-\.\/])+\.git'
,
self
.
ranRepository
)
if
result
is
not
None
:
full_ran_repo_name
=
self
.
ranRepository
.
replace
(
'git/'
,
'git'
)
else
:
full_ran_repo_name
=
self
.
ranRepository
+
'.git'
sshSession
.
command
(
'mkdir -p '
+
sourcePath
,
'\$'
,
5
)
sshSession
.
command
(
'cd '
+
sourcePath
,
'\$'
,
5
)
sshSession
.
command
(
'if [ ! -e .git ]; then stdbuf -o0 git clone '
+
full_ran_repo_name
+
' .; else stdbuf -o0 git fetch --prune; fi'
,
'\$'
,
600
)
# Raphael: here add a check if git clone or git fetch went smoothly
sshSession
.
command
(
'git config user.email "jenkins@openairinterface.org"'
,
'\$'
,
5
)
sshSession
.
command
(
'git config user.name "OAI Jenkins"'
,
'\$'
,
5
)
sshSession
.
command
(
'echo '
+
password
+
' | sudo -S git clean -x -d -ff'
,
'\$'
,
30
)
sshSession
.
command
(
'mkdir -p cmake_targets/log'
,
'\$'
,
5
)
# if the commit ID is provided use it to point to it
if
self
.
ranCommitID
!=
''
:
sshSession
.
command
(
'git checkout -f '
+
self
.
ranCommitID
,
'\$'
,
30
)
# if the branch is not develop, then it is a merge request and we need to do
# the potential merge. Note that merge conflicts should already been checked earlier
if
(
self
.
ranAllowMerge
):
if
self
.
ranTargetBranch
==
''
:
if
(
self
.
ranBranch
!=
'develop'
)
and
(
self
.
ranBranch
!=
'origin/develop'
):
sshSession
.
command
(
'git merge --ff origin/develop -m "Temporary merge for CI"'
,
'\$'
,
5
)
else
:
logging
.
debug
(
'Merging with the target branch: '
+
self
.
ranTargetBranch
)
sshSession
.
command
(
'git merge --ff origin/'
+
self
.
ranTargetBranch
+
' -m "Temporary merge for CI"'
,
'\$'
,
5
)
def
_recreate_entitlements
(
self
,
sshSession
):
# recreating entitlements, don't care if deletion fails
sshSession
.
command
(
'oc delete secret etc-pki-entitlement'
,
'\$'
,
5
)
if
re
.
search
(
r"not found"
,
sshSession
.
getBefore
()):
logging
.
warning
(
"no secrets etc-pki-entitlement found, recreating"
)
sshSession
.
command
(
'ls /etc/pki/entitlement/???????????????????.pem | tail -1'
,
'\$'
,
5
,
silent
=
True
)
regres1
=
re
.
search
(
r"/etc/pki/entitlement/[0-9]+.pem"
,
sshSession
.
getBefore
())
sshSession
.
command
(
'ls /etc/pki/entitlement/???????????????????-key.pem | tail -1'
,
'\$'
,
5
,
silent
=
True
)
regres2
=
re
.
search
(
r"/etc/pki/entitlement/[0-9]+-key.pem"
,
sshSession
.
getBefore
())
if
regres1
is
None
or
regres2
is
None
:
logging
.
error
(
"could not find entitlements"
)
return
false
file1
=
regres1
.
group
(
0
)
file2
=
regres2
.
group
(
0
)
sshSession
.
command
(
f'oc create secret generic etc-pki-entitlement --from-file
{
file1
}
--from-file
{
file2
}
'
,
'\$'
,
5
)
regres
=
re
.
search
(
r"secret/etc-pki-entitlement created"
,
sshSession
.
getBefore
())
if
regres
is
None
:
logging
.
error
(
"could not create secret/etc-pki-entitlement"
)
return
False
return
True
def
_recreate_bc
(
self
,
sshSession
,
name
,
newTag
,
filename
):
self
.
_retag_image_statement
(
sshSession
,
name
,
name
,
newTag
,
filename
)
sshSession
.
command
(
f'oc delete -f
{
filename
}
'
,
'\$'
,
5
)
sshSession
.
command
(
f'oc create -f
{
filename
}
'
,
'\$'
,
5
)
before
=
sshSession
.
getBefore
()
if
re
.
search
(
'buildconfig.build.openshift.io/[a-zA-Z\-0-9]+ created'
,
before
)
is
not
None
:
return
True
logging
.
error
(
'error while creating buildconfig: '
+
sshSession
.
getBefore
())
return
False
def
_recreate_is
(
self
,
sshSession
,
name
,
newTag
,
filename
):
sshSession
.
command
(
f'sed -i -e "s#tag: *latest#tag:
{
newTag
}
#"
{
filename
}
'
,
'\$'
,
5
)
sshSession
.
command
(
f'oc delete -f
{
filename
}
'
,
'\$'
,
5
)
sshSession
.
command
(
f'oc create -f
{
filename
}
'
,
'\$'
,
5
)
before
=
sshSession
.
getBefore
()
if
re
.
search
(
'imagestream.image.openshift.io/[a-zA-Z\-0-9]+ created'
,
before
)
is
not
None
:
return
True
logging
.
error
(
'error while creating imagestream: '
+
sshSession
.
getBefore
())
return
False
def
_start_build
(
self
,
sshSession
,
name
):
# will return "immediately" but build runs in background
# if multiple builds are started at the same time, this can take some time, however
ret
=
sshSession
.
command
(
f'oc start-build
{
name
}
--from-file=
{
self
.
eNBSourceCodePath
}
'
,
'\$'
,
300
)
before
=
sshSession
.
getBefore
()
regres
=
re
.
search
(
r'build.build.openshift.io/(?P<jobname>[a-zA-Z0-9\-]+) started'
,
str
(
before
))
if
ret
!=
0
or
before
.
count
(
'Uploading finished'
)
!=
1
or
regres
is
None
:
logging
.
error
(
"error during oc start-build: "
+
sshSession
.
getBefore
())
self
.
_delete_pod
(
sshSession
,
name
)
return
None
return
regres
.
group
(
'jobname'
)
+
'-build'
def
_delete_pod
(
self
,
sshSession
,
shortName
):
sshSession
.
command
(
f"oc get pods | grep
{
shortName
}
"
,
'\$'
,
5
)
regres
=
re
.
search
(
rf'
{
shortName
}
-[0-9]+-build'
,
sshSession
.
getBefore
())
if
regres
is
not
None
:
sshSession
.
command
(
f"oc delete pod
{
regres
.
group
(
0
)
}
"
,
'\$'
,
5
)
else
:
logging
.
warning
(
f"no pod found with name
{
shortName
}
"
)
def
_wait_build_end
(
self
,
sshSession
,
jobs
,
timeout_sec
,
check_interval_sec
=
5
):
logging
.
debug
(
f"waiting for jobs
{
jobs
}
to finish building"
)
while
timeout_sec
>
0
:
# check status
for
j
in
jobs
:
sshSession
.
command
(
f'oc get pods | grep
{
j
}
'
,
'\$'
,
5
,
silent
=
True
)
if
sshSession
.
getBefore
().
count
(
'Completed'
)
>
0
:
jobs
.
remove
(
j
)
if
sshSession
.
getBefore
().
count
(
'Error'
)
>
0
:
logging
.
error
(
f'error for job
{
j
}
: '
+
sshSession
.
getBefore
())
return
False
if
jobs
==
[]:
logging
.
debug
(
'all jobs completed'
)
return
True
time
.
sleep
(
check_interval_sec
)
timeout_sec
-=
check_interval_sec
logging
.
error
(
f"timeout while waiting for end of build of
{
jobs
}
"
)
return
False
def
_retag_image_statement
(
self
,
sshSession
,
oldImage
,
newImage
,
newTag
,
filename
):
sshSession
.
command
(
f'sed -i -e "s#
{
oldImage
}
:latest#
{
newImage
}
:
{
newTag
}
#"
{
filename
}
'
,
'\$'
,
5
)
def
_pull_image
(
self
,
sshSession
,
image
,
tag
):
sshSession
.
command
(
f'oc whoami -t | sudo podman login -u oaicicd --password-stdin https://
{
self
.
OCRegistry
}
--tls-verify=false'
,
'\$'
,
5
,
silent
=
True
)
if
sshSession
.
getBefore
().
count
(
'Login Succeeded!'
)
==
0
:
return
None
imageName
=
f'
{
self
.
OCRegistry
}{
self
.
OCProjectName
}
/
{
image
}
:
{
tag
}
'
sshSession
.
command
(
f'sudo podman pull
{
imageName
}
--tls-verify=false'
,
'\$'
,
300
)
pullResult
=
sshSession
.
getBefore
()
sshSession
.
command
(
f'sudo podman logout https://
{
self
.
OCRegistry
}
'
,
'\$'
,
10
,
silent
=
True
)
if
pullResult
.
count
(
'Storing signatures'
)
==
0
:
return
None
return
imageName
def
_get_image_size
(
self
,
sshSession
,
image
,
tag
):
# get the SHA of the image we built using the image name and its tag
sshSession
.
command
(
f'oc describe is
{
image
}
| grep -A4
{
tag
}
'
,
'\$'
,
5
)
result
=
re
.
search
(
f'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/(?P<imageSha>
{
image
}
@sha256:[a-f0-9]+)'
,
sshSession
.
getBefore
())
if
result
is
None
:
return
-
1
imageSha
=
result
.
group
(
"imageSha"
)
# retrieve the size
sshSession
.
command
(
f'oc get -o json isimage
{
imageSha
}
| jq -Mc "{{dockerImageSize: .image.dockerImageMetadata.Size}}"'
,
'\$'
,
5
)
result
=
re
.
search
(
'{"dockerImageSize":(?P<size>[0-9]+)}'
,
str
(
sshSession
.
getBefore
()))
if
result
is
None
:
return
-
1
return
int
(
result
.
group
(
"size"
))
def
_deploy_pod
(
self
,
sshSession
,
filename
,
timeout
=
30
):
sshSession
.
command
(
f'oc create -f
{
filename
}
'
,
'\$'
,
10
)
result
=
re
.
search
(
f'pod/(?P<pod>[a-zA-Z0-9_\-]+) created'
,
sshSession
.
getBefore
())
if
result
is
None
:
logging
.
error
(
f'could not deploy pod:
{
sshSession
.
getBefore
()
}
'
)
return
None
pod
=
result
.
group
(
"pod"
)
logging
.
debug
(
f'checking if pod
{
pod
}
is in Running state'
)
while
timeout
>
0
:
sshSession
.
command
(
f'oc get pod
{
pod
}
-o json | jq -Mc .status.phase'
,
'\$'
,
5
,
silent
=
True
)
if
re
.
search
(
'"Running"'
,
sshSession
.
getBefore
())
is
not
None
:
return
pod
timeout
-=
1
time
.
sleep
(
1
)
logging
.
error
(
f'pod
{
pod
}
did not reach Running state'
)
self
.
_undeploy_pod
(
sshSession
,
filename
)
return
None
def
_undeploy_pod
(
self
,
sshSession
,
filename
):
# to save time we start this in the background and trust that oc stops correctly
sshSession
.
command
(
f'oc delete -f
{
filename
}
&'
,
'\$'
,
5
)
def
BuildClusterImage
(
self
,
HTML
):
if
self
.
ranRepository
==
''
or
self
.
ranBranch
==
''
or
self
.
ranCommitID
==
''
:
HELP
.
GenericHelp
(
CONST
.
Version
)
sys
.
exit
(
f'Insufficient Parameter: ranRepository
{
self
.
ranRepository
}
ranBranch
{
ranBranch
}
ranCommitID
{
self
.
ranCommitID
}
'
)
lIpAddr
=
self
.
eNBIPAddress
lSourcePath
=
self
.
eNBSourceCodePath
if
lIpAddr
==
''
or
lSourcePath
==
''
:
HELP
.
GenericHelp
(
CONST
.
Version
)
sys
.
exit
(
'Insufficient Parameter: no SSH Credentials'
)
ocUserName
=
self
.
OCUserName
ocPassword
=
self
.
OCPassword
ocProjectName
=
self
.
OCProjectName
if
ocUserName
==
''
or
ocPassword
==
''
or
ocProjectName
==
''
:
HELP
.
GenericHelp
(
CONST
.
Version
)
sys
.
exit
(
'Insufficient Parameter: no OC Credentials'
)
if
self
.
OCRegistry
.
startswith
(
"http"
)
and
not
self
.
OCRegistry
.
endswith
(
"/"
):
sys
.
exit
(
f'ocRegistry
{
self
.
OCRegistry
}
should not start with http:// or https:// and end on a slash /'
)
logging
.
debug
(
f'Building on cluster triggered from server:
{
lIpAddr
}
'
)
mySSH
=
SSH
.
SSHConnection
()
mySSH
.
open
(
lIpAddr
,
"oaicicd"
,
"SHOULD NOT BE NECESSARY"
)
self
.
testCase_id
=
HTML
.
testCase_id
# Workaround for some servers, we need to erase completely the workspace
if
self
.
forcedWorkspaceCleanup
:
mySSH
.
command
(
f'sudo rm -Rf
{
lSourcePath
}
'
,
'\$'
,
15
)
self
.
_createWorkspace
(
mySSH
,
"SHOULD NOT BE NECESSARY"
,
lSourcePath
)
# we don't necessarily need a forced workspace cleanup, but in
# order to reduce the amount of data send to OpenShift, we
# manually delete all generated files in the workspace
mySSH
.
command
(
f'rm -rf
{
lSourcePath
}
/cmake_targets/ran_build'
,
'\$'
,
30
);
baseTag
=
'develop'
forceBaseImageBuild
=
False
imageTag
=
'develop'
if
self
.
ranAllowMerge
:
# merging MR branch into develop -> temporary image
imageTag
=
'ci-temp'
if
self
.
ranTargetBranch
==
'develop'
:
# TODO IMPORTANT: delete next line, uncomment following
mySSH
.
command
(
f'git diff HEAD..origin/develop -- cmake_targets/build_oai cmake_targets/tools/build_helper | grep --colour=never -i INDEX'
,
'\$'
,
5
)
#mySSH.command(f'git diff HEAD..origin/develop -- cmake_targets/build_oai cmake_targets/tools/build_helper docker/Dockerfile.base.rhel8.2 | grep --colour=never -i INDEX', '\$', 5)
result
=
re
.
search
(
'index'
,
mySSH
.
getBefore
())
if
result
is
not
None
:
forceBaseImageBuild
=
True
baseTag
=
'ci-temp'
else
:
forceBaseImageBuild
=
True
# logging to OC Cluster and then switch to corresponding project
mySSH
.
command
(
f'oc login -u
{
ocUserName
}
-p
{
ocPassword
}
--server
{
self
.
OCUrl
}
'
,
'\$'
,
31
)
if
mySSH
.
getBefore
().
count
(
'Login successful.'
)
==
0
:
logging
.
error
(
'
\u001B
[1m OC Cluster Login Failed
\u001B
[0m'
)
mySSH
.
close
()
HTML
.
CreateHtmlTestRow
(
'N/A'
,
'KO'
,
CONST
.
OC_LOGIN_FAIL
)
return
False
mySSH
.
command
(
f'oc project
{
ocProjectName
}
'
,
'\$'
,
30
)
if
mySSH
.
getBefore
().
count
(
f'Already on project "
{
ocProjectName
}
"'
)
==
0
and
mySSH
.
getBefore
().
count
(
f'Now using project "
{
self
.
OCProjectName
}
"'
)
==
0
:
logging
.
error
(
f'
\u001B
[1mUnable to access OC project
{
ocProjectName
}
\u001B
[0m'
)
mySSH
.
command
(
'oc logout'
,
'\$'
,
30
)
mySSH
.
close
()
HTML
.
CreateHtmlTestRow
(
'N/A'
,
'KO'
,
CONST
.
OC_PROJECT_FAIL
)
return
False
self
.
_recreate_entitlements
(
mySSH
)
status
=
True
# flag to abandon compiling if any image fails
attemptedImages
=
[]
if
forceBaseImageBuild
:
self
.
_recreate_is
(
mySSH
,
'ran-base'
,
baseTag
,
'openshift/ran-base-is.yaml'
)
self
.
_recreate_bc
(
mySSH
,
'ran-base'
,
baseTag
,
'openshift/ran-base-bc.yaml'
)
ranbase_job
=
self
.
_start_build
(
mySSH
,
'ran-base'
)
attemptedImages
+=
[
'ran-base'
]
status
=
ranbase_job
is
not
None
and
self
.
_wait_build_end
(
mySSH
,
[
ranbase_job
],
600
)
if
not
status
:
logging
.
error
(
'failure during build of ran-base'
)
mySSH
.
command
(
f'oc logs
{
ranbase_job
}
> cmake_targets/log/ran-base.log'
,
'\$'
,
10
)
# recover logs by mounting image
self
.
_retag_image_statement
(
mySSH
,
'ran-base'
,
'ran-base'
,
baseTag
,
'openshift/ran-base-log-retrieval.yaml'
)
pod
=
self
.
_deploy_pod
(
mySSH
,
'openshift/ran-base-log-retrieval.yaml'
)
if
pod
is
not
None
:
mySSH
.
command
(
f'mkdir -p cmake_targets/log/ran-base'
,
'\$'
,
5
)
mySSH
.
command
(
f'oc rsync
{
pod
}
:/oai-ran/cmake_targets/log cmake_targets/log/ran-base'
,
'\$'
,
5
)
self
.
_undeploy_pod
(
mySSH
,
'openshift/ran-base-log-retrieval.yaml'
)
else
:
status
=
False
if
status
:
self
.
_recreate_is
(
mySSH
,
'oai-physim'
,
imageTag
,
'openshift/oai-physim-is.yaml'
)
self
.
_recreate_bc
(
mySSH
,
'oai-physim'
,
imageTag
,
'openshift/oai-physim-bc.yaml'
)
self
.
_retag_image_statement
(
mySSH
,
'ran-base'
,
'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/ran-base'
,
baseTag
,
'docker/Dockerfile.phySim.rhel8.2'
)
physim_job
=
self
.
_start_build
(
mySSH
,
'oai-physim'
)
attemptedImages
+=
[
'oai-physim'
]
self
.
_recreate_is
(
mySSH
,
'ran-build'
,
imageTag
,
'openshift/ran-build-is.yaml'
)
self
.
_recreate_bc
(
mySSH
,
'ran-build'
,
imageTag
,
'openshift/ran-build-bc.yaml'
)
self
.
_retag_image_statement
(
mySSH
,
'ran-base'
,
'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/ran-base'
,
baseTag
,
'docker/Dockerfile.build.rhel8.2'
)
ranbuild_job
=
self
.
_start_build
(
mySSH
,
'ran-build'
)
attemptedImages
+=
[
'ran-build'
]
wait
=
ranbuild_job
is
not
None
and
physim_job
is
not
None
and
self
.
_wait_build_end
(
mySSH
,
[
ranbuild_job
,
physim_job
],
1200
)
if
not
wait
:
logging
.
error
(
'error during build of ranbuild_job or physim_job'
)
status
=
status
and
wait
mySSH
.
command
(
f'oc logs
{
ranbuild_job
}
> cmake_targets/log/ran-build.log'
,
'\$'
,
10
)
mySSH
.
command
(
f'oc logs
{
physim_job
}
> cmake_targets/log/oai-physim.log'
,
'\$'
,
10
)
if
status
:
self
.
_recreate_is
(
mySSH
,
'oai-enb'
,
imageTag
,
'openshift/oai-enb-is.yaml'
)
self
.
_recreate_bc
(
mySSH
,
'oai-enb'
,
imageTag
,
'openshift/oai-enb-bc.yaml'
)
self
.
_retag_image_statement
(
mySSH
,
'ran-base'
,
'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/ran-base'
,
baseTag
,
'docker/Dockerfile.eNB.rhel8.2'
)
self
.
_retag_image_statement
(
mySSH
,
'ran-build'
,
'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/ran-build'
,
imageTag
,
'docker/Dockerfile.eNB.rhel8.2'
)
enb_job
=
self
.
_start_build
(
mySSH
,
'oai-enb'
)
attemptedImages
+=
[
'oai-enb'
]
self
.
_recreate_is
(
mySSH
,
'oai-gnb'
,
imageTag
,
'openshift/oai-gnb-is.yaml'
)
self
.
_recreate_bc
(
mySSH
,
'oai-gnb'
,
imageTag
,
'openshift/oai-gnb-bc.yaml'
)
self
.
_retag_image_statement
(
mySSH
,
'ran-base'
,
'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/ran-base'
,
baseTag
,
'docker/Dockerfile.gNB.rhel8.2'
)
self
.
_retag_image_statement
(
mySSH
,
'ran-build'
,
'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/ran-build'
,
imageTag
,
'docker/Dockerfile.gNB.rhel8.2'
)
gnb_job
=
self
.
_start_build
(
mySSH
,
'oai-gnb'
)
attemptedImages
+=
[
'oai-gnb'
]
self
.
_recreate_is
(
mySSH
,
'oai-lte-ue'
,
imageTag
,
'openshift/oai-lte-ue-is.yaml'
)
self
.
_recreate_bc
(
mySSH
,
'oai-lte-ue'
,
imageTag
,
'openshift/oai-lte-ue-bc.yaml'
)
self
.
_retag_image_statement
(
mySSH
,
'ran-base'
,
'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/ran-base'
,
baseTag
,
'docker/Dockerfile.lteUE.rhel8.2'
)
self
.
_retag_image_statement
(
mySSH
,
'ran-build'
,
'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/ran-build'
,
imageTag
,
'docker/Dockerfile.lteUE.rhel8.2'
)
lteue_job
=
self
.
_start_build
(
mySSH
,
'oai-lte-ue'
)
attemptedImages
+=
[
'oai-lte-ue'
]
self
.
_recreate_is
(
mySSH
,
'oai-nr-ue'
,
imageTag
,
'openshift/oai-nr-ue-is.yaml'
)
self
.
_recreate_bc
(
mySSH
,
'oai-nr-ue'
,
imageTag
,
'openshift/oai-nr-ue-bc.yaml'
)
self
.
_retag_image_statement
(
mySSH
,
'ran-base'
,
'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/ran-base'
,
baseTag
,
'docker/Dockerfile.nrUE.rhel8.2'
)
self
.
_retag_image_statement
(
mySSH
,
'ran-build'
,
'image-registry.openshift-image-registry.svc:5000/oaicicd-ran/ran-build'
,
imageTag
,
'docker/Dockerfile.nrUE.rhel8.2'
)
nrue_job
=
self
.
_start_build
(
mySSH
,
'oai-nr-ue'
)
attemptedImages
+=
[
'oai-nr-ue'
]
wait
=
enb_job
is
not
None
and
gnb_job
is
not
None
and
lteue_job
is
not
None
and
nrue_job
is
not
None
and
self
.
_wait_build_end
(
mySSH
,
[
enb_job
,
gnb_job
,
lteue_job
,
nrue_job
],
600
)
if
not
wait
:
logging
.
error
(
'error during build of eNB/gNB/lteUE/nrUE'
)
status
=
status
and
wait
# recover logs
mySSH
.
command
(
f'oc logs
{
enb_job
}
> cmake_targets/log/oai-enb.log'
,
'\$'
,
10
)
mySSH
.
command
(
f'oc logs
{
gnb_job
}
> cmake_targets/log/oai-gnb.log'
,
'\$'
,
10
)
mySSH
.
command
(
f'oc logs
{
lteue_job
}
> cmake_targets/log/oai-lte-ue.log'
,
'\$'
,
10
)
mySSH
.
command
(
f'oc logs
{
nrue_job
}
> cmake_targets/log/oai-nr-ue.log'
,
'\$'
,
10
)
# split and analyze logs
imageSize
=
{}
for
image
in
attemptedImages
:
mySSH
.
command
(
f'mkdir -p cmake_targets/log/
{
image
}
'
,
'\$'
,
5
)
mySSH
.
command
(
f'python3 ci-scripts/docker_log_split.py --logfilename=cmake_targets/log/
{
image
}
.log'
,
'\$'
,
5
)
tag
=
imageTag
if
image
!=
'ran-base'
else
baseTag
size
=
self
.
_get_image_size
(
mySSH
,
image
,
tag
)
if
size
<=
0
:
imageSize
[
image
]
=
'unknown -- BUILD FAILED'
status
=
False
else
:
sizeMb
=
float
(
size
)
/
1000000
imageSize
[
image
]
=
f'
{
sizeMb
:.
1
f
}
Mbytes (uncompressed: ~
{
sizeMb
*
2.5
:.
1
f
}
Mbytes)'
logging
.
info
(
f'
\u001B
[1m
{
image
}
size is
{
imageSize
[
image
]
}
\u001B
[0m'
)
build_log_name
=
f'build_log_
{
self
.
testCase_id
}
'
cls_containerize
.
CopyLogsToExecutor
(
mySSH
,
lSourcePath
,
build_log_name
,
lIpAddr
,
'oaicicd'
,
'SHOULD NOT MATTER'
)
mySSH
.
command
(
'for pod in $(oc get pods | tail -n +2 | awk
\'
{print $1}
\'
); do oc delete pod ${pod}; done'
,
'\$'
,
60
)
mySSH
.
command
(
'oc logout'
,
'\$'
,
30
)
mySSH
.
close
()
# Analyze the logs
collectInfo
=
cls_containerize
.
AnalyzeBuildLogs
(
build_log_name
,
attemptedImages
,
status
)
for
img
in
collectInfo
:
for
f
in
collectInfo
[
img
]:
status
=
status
and
collectInfo
[
img
][
f
][
'status'
]
if
not
status
:
logging
.
debug
(
collectInfo
)
if
status
:
logging
.
info
(
'
\u001B
[1m Building OAI Image(s) Pass
\u001B
[0m'
)
HTML
.
CreateHtmlTestRow
(
'all'
,
'OK'
,
CONST
.
ALL_PROCESSES_OK
)
else
:
logging
.
error
(
'
\u001B
[1m Building OAI Images Failed
\u001B
[0m'
)
HTML
.
CreateHtmlTestRow
(
'all'
,
'KO'
,
CONST
.
ALL_PROCESSES_OK
)
HTML
.
CreateHtmlNextTabHeaderTestRow
(
collectInfo
,
imageSize
)
return
status
ci-scripts/main.py
View file @
41a9ac11
...
...
@@ -45,6 +45,7 @@ import cls_containerize #class Containerize for all container-based oper
import
cls_static_code_analysis
#class for static code analysis
import
cls_ci_ueinfra
#class defining the multi Ue infrastrucure
import
cls_physim1
#class PhySim for physical simulators deploy and run
import
cls_cluster
# class for building/deploying on cluster
import
sshconnection
import
epc
...
...
@@ -102,15 +103,17 @@ def AssignParams(params_dict):
def
GetParametersFromXML
(
action
):
if
action
==
'Build_eNB'
or
action
==
'Build_Image'
or
action
==
'Build_Proxy'
:
if
action
==
'Build_eNB'
or
action
==
'Build_Image'
or
action
==
'Build_Proxy'
or
action
==
"Build_Cluster_Image"
:
RAN
.
Build_eNB_args
=
test
.
findtext
(
'Build_eNB_args'
)
CONTAINERS
.
imageKind
=
test
.
findtext
(
'kind'
)
forced_workspace_cleanup
=
test
.
findtext
(
'forced_workspace_cleanup'
)
RAN
.
Build_eNB_forced_workspace_cleanup
=
False
CONTAINERS
.
forcedWorkspaceCleanup
=
False
CLUSTER
.
forcedWorkspaceCleanup
=
False
if
forced_workspace_cleanup
is
not
None
and
re
.
match
(
'true'
,
forced_workspace_cleanup
,
re
.
IGNORECASE
):
RAN
.
Build_eNB_forced_workspace_cleanup
=
True
CONTAINERS
.
forcedWorkspaceCleanup
=
True
CLUSTER
.
forcedWorkspaceCleanup
=
True
eNB_instance
=
test
.
findtext
(
'eNB_instance'
)
if
(
eNB_instance
is
None
):
RAN
.
eNB_instance
=
0
...
...
@@ -529,6 +532,7 @@ HTML = cls_oai_html.HTMLManagement()
CONTAINERS
=
cls_containerize
.
Containerize
()
SCA
=
cls_static_code_analysis
.
StaticCodeAnalysis
()
PHYSIM
=
cls_physim1
.
PhySim
()
CLUSTER
=
cls_cluster
.
Cluster
()
ldpc
=
cls_physim
.
PhySim
()
#create an instance for LDPC test using GPU or CPU build
...
...
@@ -538,7 +542,7 @@ ldpc=cls_physim.PhySim() #create an instance for LDPC test using GPU or CPU b
#-----------------------------------------------------------
import
args_parse
py_param_file_present
,
py_params
,
mode
=
args_parse
.
ArgsParse
(
sys
.
argv
,
CiTestObj
,
RAN
,
HTML
,
EPC
,
ldpc
,
CONTAINERS
,
HELP
,
SCA
,
PHYSIM
)
py_param_file_present
,
py_params
,
mode
=
args_parse
.
ArgsParse
(
sys
.
argv
,
CiTestObj
,
RAN
,
HTML
,
EPC
,
ldpc
,
CONTAINERS
,
HELP
,
SCA
,
PHYSIM
,
CLUSTER
)
...
...
@@ -926,6 +930,9 @@ elif re.match('^TesteNB$', mode, re.IGNORECASE) or re.match('^TestUE$', mode, re
HTML
=
ldpc
.
Run_NRulsimTest
(
HTML
,
CONST
,
id
)
if
ldpc
.
exitStatus
==
1
:
RAN
.
prematureExit
=
True
elif
action
==
'Build_Cluster_Image'
:
if
not
CLUSTER
.
BuildClusterImage
(
HTML
):
RAN
.
prematureExit
=
True
elif
action
==
'Build_Image'
:
CONTAINERS
.
BuildImage
(
HTML
)
elif
action
==
'Build_Proxy'
:
...
...
ci-scripts/xml_class_list.yml
View file @
41a9ac11
-
Build_Proxy
-
Build_Cluster_Image
-
Build_PhySim
-
Run_LDPCTest
-
Run_NRulsimTest
...
...
ci-scripts/xml_files/cluster_image_build.xml
0 → 100644
View file @
41a9ac11
<!--
Licensed to the OpenAirInterface (OAI) Software Alliance under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The OpenAirInterface Software Alliance licenses this file to You under
the OAI Public License, Version 1.1 (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.openairinterface.org/?page_id=698
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.
For more information about the OpenAirInterface (OAI) Software Alliance:
contact@openairinterface.org
-->
<testCaseList>
<htmlTabRef>
build-tab
</htmlTabRef>
<htmlTabName>
Build Images on Cluster
</htmlTabName>
<htmlTabIcon>
wrench
</htmlTabIcon>
<TestCaseRequestedList>
000001
</TestCaseRequestedList>
<TestCaseExclusionList></TestCaseExclusionList>
<testCase
id=
"000001"
>
<class>
Build_Cluster_Image
</class>
<desc>
Build Images on OpenShift Cluster
</desc>
</testCase>
</testCaseList>
openshift/oai-physim-bc.yaml
View file @
41a9ac11
...
...
@@ -31,10 +31,6 @@ spec:
strategy
:
dockerStrategy
:
dockerfilePath
:
"
docker/Dockerfile.phySim.rhel8.2"
from
:
kind
:
ImageStreamTag
namespace
:
oaicicd-ran
name
:
oai-ran-base:latest
output
:
to
:
kind
:
"
ImageStreamTag"
...
...
openshift/ran-base-log-retrieval.yaml
0 → 100644
View file @
41a9ac11
# Creating a pod to retrieve ran-base logs
apiVersion
:
v1
kind
:
Pod
metadata
:
name
:
ran-base-log-retrieval
namespace
:
oaicicd-ran
spec
:
containers
:
-
name
:
logcontainer
image
:
ran-base:latest
command
:
[
"
/bin/sh"
,
"
-c"
]
args
:
[
"
sleep
inf"
]
openshift/ran-build-bc.yaml
View file @
41a9ac11
...
...
@@ -31,10 +31,6 @@ spec:
strategy
:
dockerStrategy
:
dockerfilePath
:
"
docker/Dockerfile.build.rhel8.2"
from
:
kind
:
ImageStreamTag
# Use an imagestream tag as 'FROM'
namespace
:
oaicicd-ran
# Which project?
name
:
oai-ran-base:latest
# Use the imagestream created above
output
:
to
:
kind
:
"
ImageStreamTag"
...
...
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