WTS/game/scripts/utility/updater.rpy
LoafyLemon 5d9b311430 Bug fix
* Fixed updater crash caused by missing updater image during a potential server outage
2023-07-01 15:03:46 +01:00

334 lines
11 KiB
Plaintext

init python:
import requests
UPDATE_URL = bytes.fromhex("687474703a2f2f757064617465732e73696c76657273747564696f67616d65732e6f72672f757064617465732e6a736f6e").decode()
UPDATE_VER = ""
@renpy.pure
class CheckUpdates(Action):
def __init__(self, interval=3600*6, simulate=None, onetime=False, autostart=True, **kwargs):
self.url = UPDATE_URL
self.interval = interval
self.simulate = simulate
self.onetime = onetime
self.autostart = autostart
self.kwargs = kwargs
def __call__(self):
# Since source files are publicly available,
# we need to forbid updater from affecting
# GIT-supplied versions of the game to avoid bugs.
if (not updater.can_update() or config.developer) and not self.simulate:
return
check = True
url = self.url
if self.onetime and url in updater.checked:
check = False
if time.time() < persistent._update_last_checked.get(url, 0) + self.interval:
check = False
if check:
updater.checked.add(url)
persistent._update_last_checked[url] = time.time()
updater.Updater(url, check_only=True, simulate=self.simulate, **self.kwargs)
global UPDATE_VER
UPDATE_VER = persistent._update_version.get(url, "")
if self.autostart:
renpy.invoke_in_new_context(updater.update, self.url, simulate=self.simulate, **self.kwargs)
@renpy.pure
class InstallUpdates(Action):
def __init__(self, simulate=None, **kwargs):
self.url = UPDATE_URL
self.simulate = simulate
self.kwargs = kwargs
def __call__(self):
renpy.invoke_in_new_context(updater.update, self.url, simulate=self.simulate, **self.kwargs)
def fix_return_stack():
for layer in config.layers:
renpy.scene(layer)
for channel in renpy.audio.audio.channels.keys():
if isinstance(channel, str) and not channel.startswith("_"):
renpy.music.stop(channel)
renpy.set_return_stack(("main_room",))
def version_float():
control, major, *minor = config.version.split(" ")[0].split(".")
return float("{}.{}{}".format(control, major, "".join(minor)))
def version_patch():
if renpy.is_init_phase():
# Don't update save files from when game recovers from a crash.
return
latest = version_float()
# For unknown reasons, sometimes version is missing from the save, so we need a fallback
current = getattr(renpy.store, "version", latest)
# if current <= 1.431:
# # code
if current <= 1.4501:
if states.ton.unlocked and not states.ton.ev.intro.e1_complete:
states.ton.ev.intro.e1_complete = True
states.ton.ev.intro.e2_complete = True
states.ton.ev.intro.e3_complete = True
for i in states.dolls:
i = getattr(renpy.store, i)
for j in i.outfits:
j.group.sort()
j._hash = j.generate_hash()
if hasattr(states.her.ev.yule_ball, "completed"):
states.her.ev.yule_ball.complete = states.her.ev.yule_ball.completed
del states.her.ev.yule_ball.completed
if hasattr(states.her.ev.suck_it, "cock_kiss"):
states.her.ev.give_me_a_handy.cock_kiss = states.her.ev.suck_it.cock_kiss
del states.her.ev.suck_it.cock_kiss
hooch_chibi.hide()
if current <= 1.4502:
renpy.music.stop("sound2")
cho_top_school2.level = 0
cho_top_school3.level = 0
cho_outfit_bikini.desc = "It's like a regular sized bikini that's shrunk in the wash."
cho_panties_lace1.level = 14
if current > latest:
raise Exception("Loaded save file is incompatible. (Save Version: {}, Game Version: {})".format(current, latest))
if current < latest:
setattr(renpy.store, "version", latest)
message = "Have fun!"
achievements.attempt_repair()
renpy.call_in_new_context("modal_popup", "Update Successful", "\nYour save file has been successfully updated to version {{b}}{}{{/b}}.\n\n{}".format(config.version, message), None, "Hurray!")
renpy.block_rollback()
return
def version_logo():
url = UPDATE_URL
filename = "logo_{}.webp".format(UPDATE_VER)
path = os.path.join(config.basedir, "update", filename)
# Read file if exists
if os.path.isfile(path):
with open(path, "rb") as f:
data = f.read()
return im.Data(data, path)
# Fetch file if doesn't exist
url = url.split("updates.json")[0] + "logo.webp"
try:
response = requests.get(url, timeout=5)
response.raise_for_status()
data = response.content
with open(path, "wb") as f:
f.write(data)
return im.Data(data, path)
except:
path = os.path.join(config.basedir, "update", "generic.webp")
if os.path.isfile(path):
with open(path, "rb") as f:
data = f.read()
else:
data = Null()
return Fixed(im.Data(data, path), Text(UPDATE_VER, size=96, align=(0.5, 0.8), color="#000000", outlines=[( 1, "#ffffff", 0, 0 )]), fit_first=True)
config.after_load_callbacks.append(version_patch)
define update_message_list = [
"Spinning up disks",
"Lubricating Hermione",
"Filling up Tonks' glass",
"Rendering Cho's marvellous abs",
"Teaching Astoria some manners",
"Gazing into the crystal ball",
"Waxing Cho's broom",
"Sanitizing Dumbledore's desk",
"Confiscating Hermione's panties",
"Adjusting the dress code",
"Deleting Hermiobrine",
"Initializing Snape walk physics",
"Loading Genie's name mispronunciation engine",
"Generating {s}dad{/s} bad jokes",
"Perverting common nouns",
"Inserting obscure pop culture references",
"Loading wardrobe feature fondling",
"Corrupting Teachers",
"Ignoring plot holes",
"Inflating Susan's tits",
"Hiding Room of Requirement",
"Predicting images",
"Initializing sex-drive",
"Loading loading screen",
"Bribing ministry of magic employees",
"Maximizing student gullibility levels",
"Tuning out Luna's rambles",
"Filling cupboard with trash",
"Shredding rule violation reports",
"Adding tissue box of holding to desk",
"Ignoring established lore",
"Neglecting minor characters",
"Hiding the marauders map",
"Injecting British slang",
"Stiffening nipples",
"Nickering at knickers",
"Greasing Snape's hair",
"Increasing Genie's imagination levels",
"Adding illusion of choice dialogue options",
"Redirecting blood flow",
"Slutifying Slytherins",
"Dampening Genie's powers",
"Sleeving wizard cards",
"Breaking old save files",
"Loading pointless bird petting mechanics",
"Applying desk chair cushioning charm",
"Calculating semen trajectory",
"Opening spank-bank account at Gringotts",
"Scratching Genie's leg",
"Polishing Genie's wand",
"Defiling Cho's panties",
"Eliminating common sense",
"Applying lube",
"Restocking chocolate frog supply",
"Charming students",
"Breaking the fourth wall",
"Shortening skirts",
"Undressing house-elves",
"Lewdifying spells",
"Inserting sexual innuendoes",
"Searching for the g-spot",
"Adjusting refractory period levels",
"Indexing breast sizes",
"Prolonging the inevitable heat death of the universe",
"Redefining the big bang",
"Insert disc 2",
]
screen updater():
tag menu
default msg = renpy.random.choice(update_message_list)
default logo = version_logo()
use game_menu(_("Updater"), scroll="viewport"):
style_prefix "updater"
vbox:
spacing gui.pref_spacing
xfill True
button:
xalign 0.5
action OpenURL("https://www.silverstudiogames.org/")
add logo xysize(570, 324)
if u.state == u.ERROR:
text _("An error has occured:")
elif u.state == u.CHECKING:
text _("Fetching for updates.")
elif u.state == u.UPDATE_NOT_AVAILABLE:
text _("This program is up to date.")
elif u.state == u.UPDATE_AVAILABLE:
text _("[u.version] is available. Do you want to install it?")
hbox:
xalign 0.5
textbutton "Changelog" action OpenURL("https://www.silverstudiogames.org/p/changelog.html")
textbutton "Patreon" action OpenURL("https://patreon.com/silverstudiogames")
textbutton "Discord" action OpenURL("https://discord.gg/UbQeTCJ5RW")
elif u.state == u.PREPARING:
text _("Preparing to download the updates.")
elif u.state == u.DOWNLOADING:
text _("Downloading the updates.")
elif u.state == u.UNPACKING:
text _("Unpacking the updates.")
elif u.state == u.FINISHING:
text _("Finishing up.")
elif u.state == u.DONE:
text _("The updates have been installed. The program will restart.")
elif u.state == u.DONE_NO_RESTART:
text _("The updates have been installed.")
elif u.state == u.CANCELLED:
text _("The updates were cancelled.")
if u.message is not None:
text "[u.message!q]"
if u.progress is not None:
fixed:
bar value (u.progress or 0.0) range 1.0 style "updater_bar"
text "[msg]" color "#fff" size 10 yoffset 5
hbox:
xalign 0.5
if u.can_proceed:
textbutton _("Proceed") action u.proceed keysym "K_RETURN"
if u.can_cancel:
textbutton _("Cancel") action u.cancel keysym "K_ESCAPE"
timer 2.0 action SetScreenVariable("msg", renpy.random.choice(update_message_list)) repeat True
style updater_text:
xalign 0.5
style updater_button_text is gui_button_text
style updater_label is gui_label:
xsize 209
right_padding 17
style updater_label_text is gui_label_text:
xalign 1.0
text_align 1.0
outlines [(2, "#000", 0, 0)]
style updater_bar is empty:
ysize gui.bar_size
left_bar Frame("gui/bar/left.png", Borders(8, 8, 8, 8), tile=False)
right_bar Frame("gui/bar/right.png", Borders(8, 8, 8, 8), tile=False)
style update_available_button:
xalign 0.5
style update_available_button_text:
color "#F9A001"
hover_color "#fff"
xalign 0.5
label before_main_menu():
python:
if settings.get("updates") and not prerelease:
CheckUpdates(onetime=True, autostart=False)()
return