I’m trying to make code run a minute before sunrise, but since updating the code (via another question) as I’ve changed time zones from GMT, I’m having trouble getting the syntax right when removing one minute.
sunriselessonemin = (ephem.date(sunrise)) + (1*ephem.minute)
Sunrise is obtained by
sunrise, sunset = ephem.localtime(home.next_rising(sun)),ephem.localtime(home.next_setting(sun))
Can anyone help a noob?
Merci
Edited to be more clearer:-)
Here is my original code. The raspberry pi i use reboots via crontab at 4am, on starting, this script runs. It ran fine in the UK, but now im not in that time zone, i needed to add in some local timezone work as per brandons previous advice.
import sys
import os
import time
import ephem
#find time of sun rise and sunset
sun = ephem.Sun()
home = ephem.Observer()
home.lat, home.lon = '45.226691', '0.013133' #your lat long
sun.compute(home)
sunrise, sunset = ephem.localtime(home.next_rising(sun)),ephem.localtime(home.next_setting(sun))
daylightminutes = (sunset - sunrise) * 1440 # find howmany minutes of daylight there are
sunriselessonemin = ephem.date(sunrise + 1*ephem.minute)
print "prog started at(home.date) = %s" %(home.date)
print "datetime = %s" % time.strftime("%Y/%-m/%-d %H:%M:%S")
print "sunrise = %s" %sunrise
print "sunset = %s" %sunset
print "daylight mins = %s" %(daylightminutes)
testmode = "yes" #yes or no
def dostuff() :
if testmode == "yes" or sunrise <= ephem.now() <= sunriselessonemin: #if time now is within a minute of sunrise, start taking pictures
print "it's sunrise!"
if testmode == "yes" :
print "TESTMODE - Taking 10 images with 10 seconds in between and uploading made mp4 to Dropbox"
FRAMES = daylightminutes # number of images you want in timelapse video
if testmode == "yes" :
FRAMES = 10
FPS_IN = 8 # number of images per second you want in video
FPS_OUT = 8 # number of fps in finished video 24 is a good value
TIMEBETWEEN = 60 # number of seconds between pictures, 60 = 1 minute
#take the pictures needed for the time lapse video
if testmode == "yes" :
TIMEBETWEEN = 10
frameCount = 1
while frameCount < (FRAMES + 1):
print "taking image number ", frameCount, " of ", daylightminutes
datetimenowis = ephem.now()
imageNumber = str(frameCount).zfill(7)
os.system("raspistill -o /home/pi/image%s.jpg"%(imageNumber)) # -rot 270 need for cam on side (put -rot 270 before -o)
os.system("/usr/bin/convert /home/pi/image%s.jpg -pointsize 72 -fill white -annotate +40+1590 'Chicken Cam %s' /home/pi/image%s.jpg"%(imageNumber,datetimenowis,imageNumber))
frameCount += 1
time.sleep(TIMEBETWEEN - 10) #Takes roughly 6 seconds to take a picture & 4 to add text to image
#record current time and date in variable datetime
datetimenowis = time.strftime("%Y%m%d-%H%M")
print "It's sunset, processing images into one mp4 video. Time now is ", datetimenowis
# make the timelapse video out of the images taken
os.system("avconv -r %s -i /home/pi/image%s.jpg -r %s -vcodec libx264 -crf 20 -g 15 -vf crop=2592:1458,scale=1280:720 /home/pi/timelapse%s.mp4" %(FPS_IN,'%7d',FPS_OUT,datetimenowis))
#send the timelapse video to dropbox
print "Sending mp4 video to dropbox."
from subprocess import call
photofile = "/home/pi/Dropbox-Uploader/dropbox_uploader.sh upload /home/pi/timelapse%s.mp4 timelapse%s.mp4" %(datetimenowis,datetimenowis)
call ([photofile], shell=True)
print "mp4 uploaded to dropbox! Cleaning up."
#remove the timelapse video copy and all images it is made up of that are held localy on the Rpi
os.system("rm /home/pi/timelapse%s.mp4"%(datetimenowis))
os.system("rm /home/pi/image*")
print "Finished, exiting program."
sys.exit()
while ephem.now() <= sunrise:
time.sleep(1)
dostuff()
The issue at the momnet, is that if i try to tringer the main code at one minute befoe sunset. the code fails here.
pi#raspberrypi ~ $ sudo python timelapseV3.py
Traceback (most recent call last):
File "timelapseV3.py", line 13, in <module>
sunriselessonemin = ephem.date(sunrise + 1*ephem.minute)
TypeError: unsupported operand type(s) for +: 'datetime.datetime' and
'float'
I cant seem to start the code a minute before sunrise like i could before.
cheers
Once you run localtime() on a PyEphem date, you move it away from being a value that PyEphem can reason about and instead create a value that Python’s native datetime module knows about. To use the value in PyEphem, keep a copy of the raw value returned by PyEphem, and do the math with it instead:
sunrise, sunset = home.next_rising(sun), home.next_setting(sun)
sunrise_localtime, sunset_localtime = ephem.localtime(sunrise), ephem.localtime(sunset)
daylightminutes = (sunset_localtime - sunrise_localtime) * 1440 # find howmany minutes of daylight there are
sunriselessonemin = ephem.date(sunrise + 1*ephem.minute)
You should find that this runs without error. Good luck!
Related
I'm not a proficient coder, and just trying out some stuff. I have a code that kind of works, but not how I want to.
To put it simply, this is a simple downloader that asks how many files are needed, takes links and file names in order accordingly, and uses urllib.request.urlretrieve to download them. I want to show a line with the percentage downloaded, speed, duration, EST etc, and show a progress bar BELOW it in a separate line.
I tried a lot with \r carriage returns, \n new lines, and \x1B[2A ANSIs told in other replies, but I can't get my head around how to do this. My code works well when it prints in a single line, but when I try to break it into two, it either stops updating or gives me an never ending one after the other queue. Check the images below.
Here is the code that works well as far as functions are concerned, but I want this progress bar UNDER the other values.
This works, but I want this filling progress bar UNDER these values
from google.colab import drive
drive.mount("/content/drive/")
total = input("How many files? : ")
total = int(total)
link = [""]*total
name = [""]*total
for i in range (total):
print ("File No.", i+1)
link[i] = input("Enter link : ")
name[i] = input("Enter file name : ")
import urllib.request
import shutil
import sys
import time
def reporthook(count, block_size, total_size):
global start_time
if count == 0:
start_time = time.time()
return
duration = time.time() - start_time
progress_size = int(count * block_size)
speed = int(progress_size / (1024 * duration))
percent = int(count * block_size * 100 / total_size)
step = int(percent/5)
est = (total_size/speed/1024)
done = "░░"
for i in range (step):
done+="░░"
sys.stdout.write("\r%d%%, %d MB of %d MB, at %d KB/s, time elapsed = %ds, EST = %ds %s" %
(percent, progress_size / (1024 * 1024), total_size / (1024 * 1024), speed, duration, est, done))
def save(url, filename):
urllib.request.urlretrieve(url, filename, reporthook)
for j in range (total):
print("Downloading :", name[j])
save(link[j], name[j])
shutil.move(name[j], "/content/drive/My Drive/")
print("")
print(name[j], "downloaded.")
This is the code where I try to break the line into two by using a \n new line character before the "done" variable, and it becomes a never ending stream.
Never ending stream one after another
from google.colab import drive
drive.mount("/content/drive/")
total = input("How many files? : ")
total = int(total)
link = [""]*total
name = [""]*total
for i in range (total):
print ("File No.", i+1)
link[i] = input("Enter link : ")
name[i] = input("Enter file name : ")
import urllib.request
import shutil
import sys
import time
def reporthook(count, block_size, total_size):
global start_time
if count == 0:
start_time = time.time()
return
duration = time.time() - start_time
progress_size = int(count * block_size)
speed = int(progress_size / (1024 * duration))
percent = int(count * block_size * 100 / total_size)
step = int(percent/5)
est = (total_size/speed/1024)
done = "░░"
for i in range (step):
done+="░░"
sys.stdout.write("\r%d%%, %d MB of %d MB, at %d KB/s, time elapsed = %ds, EST = %ds \n%s" %
(percent, progress_size / (1024 * 1024), total_size / (1024 * 1024), speed, duration, est, done))
def save(url, filename):
urllib.request.urlretrieve(url, filename, reporthook)
for j in range (total):
print("Downloading :", name[j])
save(link[j], name[j])
shutil.move(name[j], "/content/drive/My Drive/")
print("")
print(name[j], "downloaded.")
I tried several other methods like adding multiple combinations of \r and \n at different places, \x1B[2A for ANSI ESC and everything else suggested in similar questions. All it does is either a similar result to above second code, or completely wipes out the output.
Can someone please tell me how to break the progress bar into below line without overhauling the entire code?? I think this should be a very simple thing, but can't figure it out.
(I know the whole code might look childish, made with parts taken from various places, but I'm not trying to find a coding job or worrying about neatness here. Just want to make this simple downloader work. This works perfectly, apart from the progress bar being on the same line)
I'm trying to create an uptime command for the bot e.g "the bot has been online for 10 hours 15 minutes 10 seconds"
It works the way as indented up until it hits 12am, I think it might be to do with the calculation but I'm unsure on why it would add -1day when its formatted in hours minutes seconds.
This is what I have at the moment. Any help would be appreciated :)
from datetime import datetime
import os
import time
folder = r"C:\Users\Ryan Mclaughlin\PycharmProjects\Snappingbot-V2\SnappingbotV2\text files"
filename = "uptime.txt"
fp = os.path.join(folder, filename)
with open(fp, "w") as f:
f.writelines(datetime.now().time().strftime('%H:%M:%S')) # on startup of bot write the time in a file
file = open(fp)
startup_time = (file.readline()) # reads the startup time
print(startup_time)
time.sleep(10)
time_now = datetime.now().time().strftime('%H:%M:%S') # finds time now
print(time_now)
FMT = '%H:%M:%S'
tdelta = datetime.strptime(time_now, FMT) - datetime.strptime(startup_time,FMT) # calculates the difference between the startup time and now
print(tdelta)
I had to fetch live chat on Youtube and chose to use pytchat instead of Youtube API. There's no problem to fetch the chat, but it was a bit slow.
from pytchat import LiveChat
from datetime import datetime
chat = LiveChat(video_id = "36YnV9STBqc")
k=[]
while chat.is_alive():
now2 = datetime.now()
current_time2 = now2.strftime("%H:%M:%S")
print("Current Time =", current_time2,"==========") #A
try:
data = chat.get()
now3 = datetime.now()
current_time3 = now3.strftime("%H:%M:%S")
print("Current Time =", current_time3,"~~~~~~~~~~~~~") #B
for c in data.items:
comment=f"[{c.datetime}-{c.message}]"
k.append(comment) #I need to use k later
now = datetime.now()
current_time = now.strftime("%H:%M:%S")
print(comment,"Current Time =", current_time,"++++++++++++++++")
data.tick()
except KeyboardInterrupt:
chat.terminate()
break
Below shows the output of a video with 17,649 watching: (B to A took 5 secs)
Current Time = 18:49:33 ========== #A
Current Time = 18:49:33 ~~~~~~~~~~~~~ #B
[2020-05-30 18:49:29-hi] Current Time = 18:49:33 ++++++++++++++++ #4 seconds late
[2020-05-30 18:49:32-how are you] Current Time = 18:49:36 ++++++++++++++++
Current Time = 18:49:38 ========== #A
Current Time = 18:49:38 ~~~~~~~~~~~~~ #B
[2020-05-30 18:49:32-so good] Current Time = 18:49:38 ++++++++++++++++ #6 seconds late
Below shows the output of a video with 702 watching: (B to A took at least 10 secs)
Current Time = 18:49:09 ========== #A
Current Time = 18:49:10 ~~~~~~~~~~~~~ #B
[2020-05-30 18:49:06-hellp] Current Time = 18:49:10 ++++++++++++++++
[2020-05-30 18:49:07-love the music] Current Time = 18:49:15 ++++++++++++++++
Current Time = 18:49:20 ========== #A
Current Time = 18:49:20 ~~~~~~~~~~~~~ #B
[2020-05-30 18:49:15-???] Current Time = 18:49:20 ++++++++++++++++
I assume that different watching amounts will effect the time? It's also 4 to 6 secs late to fetch every chat, is it possible to solve it? Or it's just how Pytchat works?
This is a specification.
Pytchat gets the chat in exactly the same way as the browser.
If your browser displays the time of the chat and the current time down to the second, you'll get the same results.
The response of the YouTube server is presumably affected by the number of people watching the chat and the number of people posting the chat at any given time.
It needs to be verified, but as you pointed out, I'm guessing that if a lot of chat posts are made, it's taking longer for the YouTube server to process them and return the chats that are retrieved.
(If you comment out the data.tick(), you might get a little better results.)
Use
while chat.is_alive():
data = chat.get()
for c in data.sync_items():
#c can then be formatted for ur stuff
print("Formatting stuff")
sync_items() will give you appropriate realtime chat movement.
I have made a trading expert that goes through candlesticks to check if a signal has been found then execute buy , sell orders
for i in range (len(df['Open'])) :
if some logic :
buy or sell
Now I want this expert to work on real-time data, not a historical one and I struggle with making the logic on how it'd work
what I want to do is:
Looking for the last 30 bars then make some calculations on them, then
go in a loop to check the last 2 candlesticks to see if some signal
has been found
..I want the loop to work every 4 hours since I'm working on 4h
timeframe, so with every new candlestick
I'm trying to use MetaTrader5 library
copy_rates_from_pos(
symbol, // symbol name
timeframe, // timeframe
start_pos, // initial bar index
count // number of bars
)
this code would help me find the last 30 bars but still can't get my head around on how to make the for loop !
you could use something like this
import pytz
import pandas as pd
import MetaTrader5 as mt5
import time
from datetime import datetime
from threading import Timer
server_name = "AMPGlobalUSA-Demo"
server_num = # your server num
password = # password
#------------------------------------------------------------------------------
def actualtime():
# datetime object containing current date and time
now = datetime.now()
dt_string = now.strftime("%d/%m/%Y %H:%M:%S")
#print("date and time =", dt_string)
return str(dt_string)
#------------------------------------------------------------------------------
def sync_60sec(op):
info_time_new = datetime.strptime(str(actualtime()), '%d/%m/%Y %H:%M:%S')
waiting_time = 60 - info_time_new.second
t = Timer(waiting_time, op)
t.start()
print(actualtime(), f'waiting till next minute and 00 sec...')
#------------------------------------------------------------------------------
def program(symbol):
if not mt5.initialize(login=server_num, server=server_name, password=password):
print("initialize() failed, error code =",mt5.last_error())
quit()
timezone = pytz.timezone("Etc/UTC")
utc_from = datetime.now()
######### Change here the timeframe
rates = mt5.copy_rates_from(symbol, mt5.TIMEFRAME_M1, utc_from, 70)
mt5.shutdown()
rates_frame = pd.DataFrame(rates)
rates_frame['time']=pd.to_datetime(rates_frame['time'], unit='s')
# If you want to work only with open, high, low, close you could use
#rates_frame = rates_frame.drop(['tick_volume', 'real_volume'], axis=1)
print(f"\n", actualtime(),f"|| waiting for signals {symbol} ||\n")
if not mt5.initialize():
print("initialize() failed, error code =",mt5.last_error())
quit()
point = mt5.symbol_info(symbol).point
price = mt5.symbol_info_tick(symbol).ask
request = {
"action": mt5.TRADE_ACTION_PENDING,
"symbol": symbol,
"volume": 1.0,
"type": mt5.ORDER_TYPE_BUY_LIMIT,
"price": price,
"sl": price + 40 * point,
"tp": price - 80 * point,
"deviation": 20,
"magic": 234000,
"comment": "st_1_min_mod_3",
"type_time": mt5.ORDER_TIME_GTC,
"type_filling": mt5.ORDER_FILLING_RETURN,
}
condition_buy_1 = (
(rates_frame.close.iloc[-2] > rates_frame.open.iloc[-2])&
(rates_frame.close.iloc[-2] > rates_frame.close.iloc[-3])
)
if condition_buy_1:
#result = mt5.order_send(request)
print('Sending Order!')
# Im using AMPGlobalUSA-Demo Server
# starting mt5
if not mt5.initialize(login=server_num, server=server_name, password=password):
print("initialize() failed, error code =",mt5.last_error())
quit()
#------------------------------------------------------------------------------
# S T A R T I N G M T 5
#------------------------------------------------------------------------------
authorized=mt5.login(server_num, password=password)
if authorized:
account_info=mt5.account_info()
if account_info!=None:
account_info_dict = mt5.account_info()._asdict()
df=pd.DataFrame(list(account_info_dict.items()),columns=['property','value'])
print("account_info() as dataframe:")
print(df)
else:
print(f"failed to connect to trade account {server_num} with password={password}, error code =",mt5.last_error())
mt5.shutdown()
#------------------------------------------------------------------------------
def trading_bot():
symbol_1 = 'EURUSD'
symbol_2 = 'EURCAD'
while True:
program(symbol_1)
program(symbol_2)
time.sleep(59.8) # it depends on your computer and ping
sync_60sec(trading_bot)
Here you have the basics how to connect and operate with Python and MT5. You can save it as py file. And you have the first script looking for signals in 1 min chart for your symbol. You could have two different scripts mor looking for signals in 5 and 15 min charts, (program_5min.py and program_15min.py). You should then, add a new sync function. For example, for 5 minutes you have to wait for one hour and 0,5,10,15 minutes and so one.
Hope it works for you, have fun!
I need to make a system where the user can set a timer that goes off every x hours for x minutes (ideally starting from 0600 or at least on the 00 seconds of the actual time) over 24 hours
below is an example of the code i am trying to do, the problem is when it hits 00:00 (midnight) it goes back to being not ready.
import time
import argparse
from datetime import datetime
def main(start_time_hour, start_time_minute, watering_duration,
watering_interval_hour, watering_interval_minute):
# Watering loop, each cycle consists one application of water per watering
cycle.
while True:
# Sleep until start time is reached. Resets every 24 hours.
current_time = datetime.now()
while current_time.hour < start_time_hour: # This loop will only break after start hour is reached,
time.sleep(1) # at which point it will only pass for remainder of 24h
current_time = datetime.now()
print('Not Ready - Hour')
while current_time.minute < start_time_minute: # This loop will only break after start minute is reached,
time.sleep(1) # at which point the watering routine will execute for
current_time = datetime.now() # the remainder of the 24 hour period
print('Not Ready - minute')
while current_time.hour <= start_time_hour and current_time.minute < start_time_minute: # This loop will only break after start hour is reached,
time.sleep(1) # at which point it will only pass for remainder of 24h
current_time = datetime.now()
print('Not Ready - minute')
# Commence watering loop
print("call function to turn sprinkler valve relay ON here")
print('On time:', datetime.now().time().strftime('%H:%M:%S'))
time.sleep(watering_duration)
print("call function to turn sprinkler valve relay OFF here")
print('Off time:', datetime.now().time().strftime('%H:%M:%S'))
time.sleep(watering_interval_hour * 3600 + watering_interval_minute * 60 - watering_duration)
if __name__ == '__main__':
parser = argparse.ArgumentParser(description="Watering timer code. Takes user input args and prints events to stdout")
parser.add_argument('start_time', metavar='start_time %s', help='start time in HHMM format with leading zeroes')
parser.add_argument('watering_duration', metavar='watering_duration %s', help='watering duration in seconds')
parser.add_argument('watering_interval', metavar='watering_interval %s', help='watering interval in HHMM format with leading zeroes')
args = parser.parse_args()
main(int(args.start_time[0:2]), int(args.start_time[2:4]), int(args.watering_duration),
int(args.watering_interval[0:2]), int(args.watering_interval[2:4]))
Thanks in Advance!
Linux crontab is a good option for the automated scheduler Job.
The crontab is a list of commands that you want to run on a regular schedule,
and also the user-defined functions/program it will be executed within the regular interval of time.
Usage:
MIN HOUR DOM MON DOW CMD
* * * * * /usr/local/bin/action-trigger.sh
MIN : Minute field 0 to 59
HOUR : Hour field 0 to 23
DOM : Day of Month 1-31
MON : Month field 1-12
DOW : Day Of Week 0-6
CMD : Command Any command to be executed