Skip to content

Commit

Permalink
Merge pull request #93 from MikeGawi/google_limit
Browse files Browse the repository at this point in the history
Preventing Google Photos API limit from being exceeded #92
  • Loading branch information
MikeGawi authored Jan 1, 2024
2 parents 7b38b06 + 4288442 commit 7a0c801
Show file tree
Hide file tree
Showing 23 changed files with 254 additions and 91 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -144,3 +144,4 @@ dmypy.json
/photo_thumb.jpg
/gen_cc_badge.py
/webtest.py
/list.fth
6 changes: 6 additions & 0 deletions config.cfg
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,12 @@ pickle_file=token.pickle
# Empty means all. Default: empty
album_names=

# Refresh rate for data from the Google Photos API.
# Possible values: always = every frame refresh, once = once a day.
# This is to prevent a large number of photos from hitting the daily API limit (10k calls).
# Default: always
refresh_rate=always

# Set 1 to get photos from local storage.
; The rest of the values in this section regarding local should be filled as well when enabled.
# Default: 1 (enabled)
Expand Down
170 changes: 121 additions & 49 deletions ePiframe.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,16 @@
import shutil
import signal
import sys
from datetime import datetime
import pandas
import starlette.status
from pandas import DataFrame
from requests import HTTPError

from misc.connection import Connection
from misc.constants import Constants
from misc.logs import Logs
from misc.tools import Tools
from modules.albummanager import AlbumManager
from modules.base.pluginbase import PluginBase
from modules.configmanager import ConfigManager
Expand Down Expand Up @@ -104,11 +111,7 @@ def process_flow(self):
)
self.logging.log(f"Photo to show:\n{photo}")

self.index_manager.set_id(
self.photo_manager.get_photo_attribute(
photo, Constants.GOOGLE_PHOTOS_ALBUMS_PHOTO_ID_HEADER
)
)
self.index_manager.set_id(self.get_photo_id(photo))
self.remove_old_files()

self.logging.log("Getting next photo...")
Expand All @@ -120,18 +123,26 @@ def process_flow(self):
returned_value,
)

def get_photo_id(self, photo):
return self.photo_manager.get_photo_attribute(
photo, Constants.GOOGLE_PHOTOS_ALBUMS_PHOTO_ID_HEADER
)

def get_next_photo(self, photo) -> tuple:
returned_value = None
if photo[Constants.PHOTOS_SOURCE_LABEL] == Constants.GOOGLE_PHOTOS_SOURCE:
filename = self.get_photo(photo)
download_url = self.get_download_url(photo)
returned_value = self.try_download_file(download_url, filename)
returned_value = self.try_download_file(
self.get_download_url(photo), self.get_photo_id(photo), filename
)
else:
plugin_with_source = self.get_plugin_with_source(photo)
filename = self.get_filename(photo, plugin_with_source, returned_value)
return filename, returned_value

def get_filename(self, photo, plugin_with_source: PluginBase, returned_value) -> str:
def get_filename(
self, photo, plugin_with_source: PluginBase, returned_value
) -> str:
if plugin_with_source and self.plugins_manager.plugin_source_get_file(
plugin_with_source
):
Expand Down Expand Up @@ -172,21 +183,42 @@ def get_plugin_with_source(self, photo) -> PluginBase:
None,
)

def try_download_file(self, download_url: str, filename: str) -> str:
try:
returned_value = Connection.download_file(
download_url,
self.config.get("photo_convert_path"),
filename,
Constants.OK_STATUS_ERRORCODE,
Constants.CHECK_CONNECTION_TIMEOUT,
)
except Exception as exception:
returned_value = str(exception)
def try_download_file(self, download_url: str, photo_id: str, filename: str) -> str:
returned_value = self.download_or_retry(download_url, filename, photo_id)

if returned_value != Constants.OK_STATUS_ERRORCODE:
self.logging.log(f"Fail! Server error: {str(returned_value)}")
return returned_value

def download_or_retry(self, download_url, filename, photo_id):
returned_value = None
count = 2
while not returned_value:
try:
returned_value = Connection.download_file(
download_url,
self.config.get("photo_convert_path"),
filename,
Constants.OK_STATUS_ERRORCODE,
Constants.CHECK_CONNECTION_TIMEOUT,
)
except HTTPError as exception:
if self.is_403_exception(exception) and count:
download_url = self.get_download_url(
self.auth_manager.get_item(photo_id)
)
count -= 1
continue
returned_value = str(exception)
return returned_value

@staticmethod
def is_403_exception(exception):
return (
hasattr(exception, "response")
and exception.response.status_code == starlette.status.HTTP_403_FORBIDDEN
)

def get_download_url(self, photo) -> str:
return (
self.photo_manager.get_photo_attribute(
Expand Down Expand Up @@ -346,38 +378,73 @@ def exit_if_no_photos(self, total_number: int):
self.logging.log("No photos in albums!")
sys.exit(1)

def get_from_sources(self):
photos = None
def get_from_sources(self) -> DataFrame:
photos: DataFrame = DataFrame()
if bool(self.config.getint("use_google_photos")):
self.logging.log("Getting data from Google Photos source...")
self.logging.log("Checking connection...")
self.check_connection()
self.logging.log("OK!")
photos = self.get_google_photos()
photos = self.get_local_source(photos)
photos = self.get_plugin_sources(photos)
return photos

self.logging.log("Loading credentials...")
self.create_auth_manager()
self.logging.log("Success!")
@staticmethod
def should_data_be_refreshed(filename: str) -> bool:
if not os.path.exists(filename):
return True

self.logging.log("Trying to build service with given credentials...")
self.build_service()
mod_time = Tools.get_last_date(filename)
if not mod_time:
return True

self.logging.log("Success!")
self.logging.log("Getting all albums...")
self.get_albums_data()
self.logging.log("Success!")
return datetime.now().date() > datetime.fromtimestamp(int(mod_time)).date()

self.logging.log("Getting desired album(s)...")
self.album_manager = AlbumManager(
self.auth_manager.get_response(),
self.config.get("album_names"),
Constants.GOOGLE_PHOTOS_ALBUMS_TITLE_HEADER,
)
def read_stored_photos(self):
photos = DataFrame()
filename: str = self.config.get("photo_list_file") + ".fth"
should_refresh: bool = self.should_data_be_refreshed(filename)
if (
self.config.get("refresh_rate") == Constants.REFRESH_ONCE
and not self.check_arguments("--refresh")
and not should_refresh
):
try:
self.logging.log(
"Trying to read saved Google Photos data (according to refresh_rate setting set to "
"'once' a day)..."
)
photos = pandas.read_feather(filename)
self.logging.log("Success!")
except Exception:
pass

photos = self.get_albums()
photos = self.get_local_source(photos)
photos = self.get_plugin_sources(photos)
return photos

def get_google_photos(self) -> DataFrame:
self.logging.log("Getting data from Google Photos source...")
self.logging.log("Checking connection...")
self.check_connection()
self.logging.log("OK!")
self.logging.log("Loading credentials...")
self.create_auth_manager()
self.logging.log("Success!")
self.logging.log("Trying to build service with given credentials...")
self.build_service()
self.logging.log("Success!")

photos = self.read_stored_photos()
if not photos.empty:
return photos

self.logging.log("Getting all albums...")
self.get_albums_data()
self.logging.log("Success!")
self.logging.log("Getting desired album(s)...")
self.album_manager = AlbumManager(
self.auth_manager.get_response(),
self.config.get("album_names"),
Constants.GOOGLE_PHOTOS_ALBUMS_TITLE_HEADER,
)
return self.get_albums()

def process_test_convert(self):
if self.check_arguments("--test-convert"):
self.test_convert()
Expand Down Expand Up @@ -445,7 +512,7 @@ def check_all_sources(self):
self.logging.log("No photo sources picked! Check the configuration!")
raise Exception("No photo sources picked! Check the configuration!")

def get_albums(self):
def get_albums(self) -> DataFrame:
if self.album_manager.get_albums().empty:
self.logging.log(
"Fail! Can't find album {}".format(self.config.get("album_names"))
Expand Down Expand Up @@ -476,11 +543,11 @@ def get_albums(self):
raise
return self.get_google_source()

def get_google_source(self):
def get_google_source(self) -> DataFrame:
if self.album_manager.get_data().empty:
self.logging.log("Fail! Couldn't retrieve albums!")
raise
photos = self.photo_manager.set_photos(
photos: DataFrame = self.photo_manager.set_photos(
self.album_manager,
Constants.GOOGLE_PHOTOS_ALBUMS_MEDIAMETADATA_HEADER,
Constants.GOOGLE_PHOTOS_ALBUMS_PHOTO_HEADER,
Expand All @@ -490,9 +557,11 @@ def get_google_source(self):
Constants.GOOGLE_PHOTOS_SOURCE,
)
self.logging.log("Success!")
if not photos.empty:
photos.to_feather(self.config.get("photo_list_file") + ".fth")
return photos

def get_local_source(self, photos):
def get_local_source(self, photos: DataFrame) -> DataFrame:
if bool(self.config.getint("use_local")):
self.logging.log("Getting data from local source...")
self.local_source_manager = LocalSourceManager(
Expand All @@ -510,7 +579,7 @@ def get_local_source(self, photos):
self.logging.log("Success!")
return photos

def get_plugin_sources(self, photos):
def get_plugin_sources(self, photos: DataFrame) -> DataFrame:
for plugin in self.plugins_manager.plugin_source():
try:
self.logging.log(f"Getting data from plugin '{plugin.name}' source...")
Expand Down Expand Up @@ -631,8 +700,11 @@ def show_help(self):
print("--test-display [file] displays the photo file on attached display")
print(" with current ePiframe configuration")
print("--test-convert [file] converts the photo file to configured")
print(" photo_convert_filename current ePiframe configuration")
print(" photo_convert_filename with current ePiframe configuration")
print("--no-skip like --test but is not skipping to another photo")
print(
"--refresh force Google API data refresh even if refresh_rate flag is set to 'once'"
)
print("--users manage users")
print("--help this help")

Expand Down
2 changes: 1 addition & 1 deletion install.sh
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ function install_pips {
echo -e '\n\033[0;30mInstalling Python components\033[0m'
declare -A pips=( ["Requests"]="requests>=2.26.0" ["Pillow"]="pillow==9.3.0" ["Telebot"]="pyTelegramBotAPI" ["Dateutil"]="python-dateutil" ["ConfigParser"]="configparser>=5.0.0"\
["Google components"]="google-api-python-client google-auth-httplib2 google-auth-oauthlib"\
["SPI Libs"]="spidev==3.5" ["Pandas"]="pandas==1.2.0 numpy==1.20" ["Flask"]="flask<2.2.0" ["Flask-WTF"]="flask-wtf==1.0.0" \
["SPI Libs"]="spidev==3.5" ["Pandas"]="pandas==1.2.0 numpy==1.20 pyarrow" ["Flask"]="flask<2.2.0" ["Flask-WTF"]="flask-wtf==1.0.0" \
["Flask-Login"]="flask-login==0.5.0" ["WTForms"]="wtforms>=3.0.0" ["SMBus"]="smbus2")
declare -a order;
order+=( "Requests" )
Expand Down
6 changes: 6 additions & 0 deletions misc/config.default
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,12 @@ pickle_file=token.pickle
# Empty means all. Default: empty
album_names=

# Refresh rate for data from the Google Photos API.
# Possible values: always = every frame refresh, once = once a day.
# This is to prevent a large number of photos from hitting the daily API limit (10k calls).
# Default: always
refresh_rate=always

# Set 1 to get photos from local storage.
; The rest of the values in this section regarding local should be filled as well when enabled.
# Default: 1 (enabled)
Expand Down
1 change: 0 additions & 1 deletion misc/connection.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
import os
import requests
import re
import socket


class Connection:
Expand Down
6 changes: 5 additions & 1 deletion misc/constants.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@

class Constants:

EPIFRAME_VERSION = "v1.9.3"
EPIFRAME_VERSION = "v1.9.4"
EPIFRAME_SECRET = "ePiframeSecretlyLovesYourPhotos"

# minimal needed python version
Expand Down Expand Up @@ -39,6 +39,7 @@ class Constants:
GOOGLE_PHOTOS_ALBUMS_PHOTO_DESCRIPTION_HEADER = "description"
GOOGLE_PHOTOS_ALBUMS_PHOTO_BASEURL_HEADER = "baseUrl"
GOOGLE_PHOTOS_ALBUMS_PHOTO_ID_HEADER = "id"
GOOGLE_PHOTOS_ALBUMS_MEDIA_ITEM_ID_HEADER = "mediaItemId"
GOOGLE_PHOTOS_ALBUMS_PHOTO_HEIGHT_HEADER = "height"
GOOGLE_PHOTOS_ALBUMS_PHOTO_WIDTH_HEADER = "width"
GOOGLE_PHOTOS_ALBUMS_PHOTO_GET_DETAILS = "=d"
Expand Down Expand Up @@ -146,3 +147,6 @@ class Constants:
BACK_BLACK = "black"
BACK_PHOTO = "photo"
BACK_CROP = "crop"

REFRESH_ALWAYS = "always"
REFRESH_ONCE = "once"
2 changes: 1 addition & 1 deletion misc/pimoronidisplay.py
Original file line number Diff line number Diff line change
Expand Up @@ -65,7 +65,7 @@ def __init_inky_what(self, color_schema: str, module: ModuleType):
)
self.__palette_filter = self.__get_palette(module)

def __init_inky_uc(self, color_schema: str, module: ModuleType):
def __init_inky_uc(self, _color_schema: str, module: ModuleType):
self.__inky = module.Inky()
self.__palette_filter = module.DESATURATED_PALETTE

Expand Down
14 changes: 13 additions & 1 deletion misc/tools.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import itertools
import os
from typing import Any


Expand All @@ -9,7 +10,9 @@ def check_if_list(key: str, value: str) -> Any:

@staticmethod
def get_type_to_extension(extensions: dict) -> dict:
return dict(Tools.check_if_list(key, value) for key, value in extensions.items())
return dict(
Tools.check_if_list(key, value) for key, value in extensions.items()
)

@staticmethod
def get_product(key: str, value: str):
Expand Down Expand Up @@ -39,3 +42,12 @@ def get_extensions(extensions: list) -> list:
[Tools.get_extension(extension) for extension in list(extensions)],
[],
)

@staticmethod
def get_last_date(file: str) -> float:
return_value = None
try:
return_value = os.stat(file).st_mtime
except Exception:
pass
return return_value
Loading

0 comments on commit 7a0c801

Please sign in to comment.