You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
201 lines
6.6 KiB
201 lines
6.6 KiB
2 years ago
|
#!/usr/bin/env python
|
||
|
|
||
|
# @HEADER
|
||
|
# ************************************************************************
|
||
|
#
|
||
|
# TriBITS: Tribal Build, Integrate, and Test System
|
||
|
# Copyright 2013 Sandia Corporation
|
||
|
#
|
||
|
# Under the terms of Contract DE-AC04-94AL85000 with Sandia Corporation,
|
||
|
# the U.S. Government retains certain rights in this software.
|
||
|
#
|
||
|
# Redistribution and use in source and binary forms, with or without
|
||
|
# modification, are permitted provided that the following conditions are
|
||
|
# met:
|
||
|
#
|
||
|
# 1. Redistributions of source code must retain the above copyright
|
||
|
# notice, this list of conditions and the following disclaimer.
|
||
|
#
|
||
|
# 2. Redistributions in binary form must reproduce the above copyright
|
||
|
# notice, this list of conditions and the following disclaimer in the
|
||
|
# documentation and/or other materials provided with the distribution.
|
||
|
#
|
||
|
# 3. Neither the name of the Corporation nor the names of the
|
||
|
# contributors may be used to endorse or promote products derived from
|
||
|
# this software without specific prior written permission.
|
||
|
#
|
||
|
# THIS SOFTWARE IS PROVIDED BY SANDIA CORPORATION "AS IS" AND ANY
|
||
|
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||
|
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL SANDIA CORPORATION OR THE
|
||
|
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
||
|
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
||
|
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
||
|
# PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||
|
# LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||
|
# NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||
|
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
#
|
||
|
# ************************************************************************
|
||
|
# @HEADER
|
||
|
|
||
|
|
||
|
# TODO: Put a time limit on the command so that it will not run past the
|
||
|
# RUNTILL time. You could do this by running a cmake -P driver script that
|
||
|
# calls EXECUTE_PROCESS(...). There is also some python code that exists out
|
||
|
# there that could also implement a time limit but it is complex.
|
||
|
|
||
|
|
||
|
#
|
||
|
# Read in the command-line arguments
|
||
|
#
|
||
|
|
||
|
usageHelp = r"""generic-looping-demon.py [OPTIONS]
|
||
|
|
||
|
This simple program takes a command as input and runs it over and over again
|
||
|
(pausing in-between iterations for a given time) and then stops at the given
|
||
|
absolute time.
|
||
|
|
||
|
The reason that the script takes an absolute time instead of a relative time
|
||
|
is that this script is desiged to drive continuous itegration (CI) processes
|
||
|
where the CI process should shut down at some point.
|
||
|
|
||
|
NOTE: The last iteration will not start later than --run-till=RUNTILL but the
|
||
|
last command could run until significantly after that time.
|
||
|
"""
|
||
|
|
||
|
from optparse import OptionParser
|
||
|
|
||
|
clp = OptionParser(usage=usageHelp)
|
||
|
|
||
|
clp.add_option(
|
||
|
"--command", dest="command", type="string", default="",
|
||
|
help="The shell command that will get run in the loop" )
|
||
|
|
||
|
clp.add_option(
|
||
|
"--loop-interval", dest="loopInterval", type="string", default="",
|
||
|
help="Input to the standard unix/linux 'sleep' command" \
|
||
|
+" (e.g. '60s') to space out iterations of the script.")
|
||
|
|
||
|
dateFormat = "%Y-%m-%d"
|
||
|
timeFormat = "%H:%M:%S"
|
||
|
dateTimeFormat = dateFormat+" "+timeFormat
|
||
|
clp.add_option(
|
||
|
"--run-till", dest="runTill", type="string", default="",
|
||
|
help="The absolute time the script will run iterations till." \
|
||
|
+" This takes the format "+dateTimeFormat+" (e.g. 2011-06-09 18:00:00).")
|
||
|
|
||
|
clp.add_option(
|
||
|
"--today-run-till", dest="todayRunTill", type="string", default="",
|
||
|
help="The time today the script will run iterations till." \
|
||
|
+" This takes the format "+timeFormat+" (e.g. 18:00:00).")
|
||
|
|
||
|
clp.add_option(
|
||
|
"--pause-file", dest="pauseFile", type="string", default="",
|
||
|
help="The name of a file, that if exists, will prevent command" \
|
||
|
+" from being run.")
|
||
|
|
||
|
|
||
|
(options, args) = clp.parse_args()
|
||
|
|
||
|
|
||
|
#
|
||
|
# Check the input arguments
|
||
|
#
|
||
|
|
||
|
import sys
|
||
|
|
||
|
if not options.command:
|
||
|
print("\nError, you must set the --command argument!")
|
||
|
sys.exit(1)
|
||
|
|
||
|
if not options.loopInterval:
|
||
|
print("\nError, you must set the --loop-interval argument!")
|
||
|
sys.exit(1)
|
||
|
|
||
|
if not (options.runTill or options.todayRunTill):
|
||
|
print("\nError, you must set either the --run-till or --today-run-till argument!")
|
||
|
sys.exit(1)
|
||
|
|
||
|
|
||
|
#
|
||
|
# Echo the command-line
|
||
|
#
|
||
|
|
||
|
print("")
|
||
|
print("**************************************************************************")
|
||
|
print("Script: generic-looping-demon.py \\")
|
||
|
print(" --command='" + options.command + "' \\")
|
||
|
print(" --loop-interval='" + options.loopInterval + "' \\")
|
||
|
|
||
|
if options.runTill:
|
||
|
print(" --run-till='" + options.runTill + "' \\")
|
||
|
if options.todayRunTill:
|
||
|
print(" --run-till='" + options.todayRunTill + "' \\")
|
||
|
print(" --pause-file='" + options.pauseFile + "' \\")
|
||
|
|
||
|
|
||
|
#
|
||
|
# Helper functions
|
||
|
#
|
||
|
|
||
|
# Parses "%Y-%m-%d %H:%M:%S" into a datetime.datetime object
|
||
|
def parseDateTimeString(dateTimeStr):
|
||
|
(dateStr, timeStr) = dateTimeStr.split(" ")
|
||
|
(year, month, day) = dateStr.split("-")
|
||
|
(hour, minute, second) = timeStr.split(":")
|
||
|
return datetime.datetime(int(year), int(month), int(day),
|
||
|
int(hour), int(minute), int(second))
|
||
|
|
||
|
|
||
|
def formatDateTime(dateTimeObj):
|
||
|
return datetime.datetime.strftime(dateTimeObj, dateTimeFormat)
|
||
|
|
||
|
|
||
|
def pauseFileExists(pauseFile):
|
||
|
if pauseFile and os.path.exists(pauseFile):
|
||
|
return True
|
||
|
return False
|
||
|
|
||
|
#
|
||
|
# Executable statements
|
||
|
#
|
||
|
|
||
|
import os
|
||
|
import datetime
|
||
|
|
||
|
scriptsDir = os.path.abspath(os.path.dirname(sys.argv[0]))+"/cmake/python"
|
||
|
sys.path.insert(0, scriptsDir)
|
||
|
|
||
|
from GeneralScriptSupport import *
|
||
|
|
||
|
if options.runTill:
|
||
|
finalDateTime = parseDateTimeString(options.runTill)
|
||
|
elif options.todayRunTill:
|
||
|
todayDate = datetime.datetime.today()
|
||
|
todayDateStr = datetime.datetime.strftime(todayDate, dateFormat)
|
||
|
finalDateTime = parseDateTimeString(todayDateStr+" "+options.todayRunTill)
|
||
|
|
||
|
if pauseFileExists(options.pauseFile):
|
||
|
print("\nThe file " + options.pauseFile + " exists at start so deleting it!")
|
||
|
os.remove(options.pauseFile)
|
||
|
|
||
|
print("\nThe script will run iterations till = " + formatDateTime(finalDateTime)
|
||
|
+ "\n")
|
||
|
|
||
|
currentTime = datetime.datetime.now()
|
||
|
iteration = 0
|
||
|
|
||
|
while currentTime < finalDateTime:
|
||
|
print("*********************************************************************"
|
||
|
"********\n" + str(iteration) + ":" + " current time = " +
|
||
|
formatDateTime(currentTime) + ", final time = " +
|
||
|
formatDateTime(finalDateTime))
|
||
|
if pauseFileExists(options.pauseFile):
|
||
|
print("\nThe file " + options.pauseFile + " exists so skipping this iteration!")
|
||
|
else:
|
||
|
echoRunSysCmnd(options.command, throwExcept=False, timeCmnd=True)
|
||
|
echoRunSysCmnd("sleep "+options.loopInterval)
|
||
|
currentTime = datetime.datetime.now()
|
||
|
iteration += 1
|