Module:AutoUpdateServPrices: Difference between revisions
From MicrasWiki
Jump to navigationJump to search
NewZimiaGov (talk | contribs) (Created page with "local p = {} local function getCurrentDate() local startDate = os.time({year = 1999, month = 8, day = 6}) local secondsInDay = 86400 local daysPerYear = 183 local currentDate = os.time() local totalDaysElapsed = math.floor((currentDate - startDate) / secondsInDay) local yearFraction = totalDaysElapsed / daysPerYear local psscYear = math.floor(yearFraction) local dayOfYear = math.floor((yearFraction - psscYear) * daysPerYear) + 1 lo...") |
NewZimiaGov (talk | contribs) No edit summary |
||
(3 intermediate revisions by the same user not shown) | |||
Line 1: | Line 1: | ||
--------------------------------------------------------- | |||
-- Module:DailyPriceTable | |||
-- Generates a daily-updated table where each cell | |||
-- refreshes once per day, with random fluctuations | |||
-- stable for that day but new on subsequent days. | |||
--------------------------------------------------------- | |||
local p = {} | local p = {} | ||
--------------------------------------------------------- | |||
-- getCurrentDate | |||
-- Returns today's date in PSSC format, | |||
-- along with dayOfMonth and month for daily logic. | |||
--------------------------------------------------------- | |||
local function getCurrentDate() | local function getCurrentDate() | ||
local startDate = os.time({year = 1999, month = 8, day = 6}) | local startDate = os.time({year = 1999, month = 8, day = 6}) | ||
Line 28: | Line 39: | ||
end | end | ||
--------------------------------------------------------- | |||
-- Product data remains the same | |||
--------------------------------------------------------- | |||
local products = { | local products = { | ||
{"Currency", "Lake Morovia Blockade Fund", 4.05}, | {"Currency", "Lake Morovia Blockade Fund", 4.05, "stacks"}, | ||
{"Plunder", "Lake Morovia Blockade Fund", 3.30}, | {"Plunder", "Lake Morovia Blockade Fund", 3.30, "chests"}, | ||
{"Prostitutes", "Court of the Dark Harpy", 0.15}, | {"Prostitutes", "Court of the Dark Harpy", 0.15, "contracts"}, | ||
{"Lievs", "Hatch Ministry", 0.32}, | {"Lievs", "Hatch Ministry", 0.32, "units"}, | ||
{"Prisoners", "Hatch Ministry", 0.04}, | {"Prisoners", "Hatch Ministry", 0.04, "cells"}, | ||
{"Faces", "Maritime Guild of the Cult of Maskmakers", 0.03}, | {"Faces", "Maritime Guild of the Cult of Maskmakers", 0.03, "masks"}, | ||
{"Spies", "Ergonian Spy Agency", 0.01}, | {"Spies", "Ergonian Spy Agency", 0.01, "assignments"}, | ||
{"Missionaries of the Order Aurora Mystica", "Temple Bank of the Reformed Stripping Path", 0.05}, | {"Missionaries of the Order Aurora Mystica", "Temple Bank of the Reformed Stripping Path", 0.05, "teams"}, | ||
{"Missionaries of Harmony Sanctum", "Temple Bank of the Reformed Stripping Path", 0.09}, | {"Missionaries of Harmony Sanctum", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, | ||
{"Missionaries of Ignis Aeternum", "Temple Bank of the Reformed Stripping Path", 0.09}, | {"Missionaries of Ignis Aeternum", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, | ||
{"Missionaries of Celestial Harmony Sect", "Temple Bank of the Reformed Stripping Path", 0.08}, | {"Missionaries of Celestial Harmony Sect", "Temple Bank of the Reformed Stripping Path", 0.08, "teams"}, | ||
{"Missionaries of the Guild of Golden Shadows", "Temple Bank of the Reformed Stripping Path", 0.05}, | {"Missionaries of the Guild of Golden Shadows", "Temple Bank of the Reformed Stripping Path", 0.05, "teams"}, | ||
{"Missionaries of the Azure Sentinel Sect", "Temple Bank of the Reformed Stripping Path", 0.01}, | {"Missionaries of the Azure Sentinel Sect", "Temple Bank of the Reformed Stripping Path", 0.01, "teams"}, | ||
{"Missionaries of Reverie Nebulous", "Temple Bank of the Reformed Stripping Path", 0.10}, | {"Missionaries of Reverie Nebulous", "Temple Bank of the Reformed Stripping Path", 0.10, "teams"}, | ||
{"Missionaries of the Eon Fellowship", "Temple Bank of the Reformed Stripping Path", 0.09}, | {"Missionaries of the Eon Fellowship", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, | ||
{"Missionaries of the Order of the Umbral Oracle", "Temple Bank of the Reformed Stripping Path", 0.07}, | {"Missionaries of the Order of the Umbral Oracle", "Temple Bank of the Reformed Stripping Path", 0.07, "teams"}, | ||
{"Missionaries of the Mystery of the Verdant Embrace", "Temple Bank of the Reformed Stripping Path", 0.10}, | {"Missionaries of the Mystery of the Verdant Embrace", "Temple Bank of the Reformed Stripping Path", 0.10, "teams"}, | ||
{"Missionaries of Conclace Illuminara", "Temple Bank of the Reformed Stripping Path", 0.02}, | {"Missionaries of Conclace Illuminara", "Temple Bank of the Reformed Stripping Path", 0.02, "teams"}, | ||
{"Missionaries of Sanctum Vitalis", "Temple Bank of the Reformed Stripping Path", 0.06}, | {"Missionaries of Sanctum Vitalis", "Temple Bank of the Reformed Stripping Path", 0.06, "teams"}, | ||
{"Missionaries of Temple Alabaster", "Temple Bank of the Reformed Stripping Path", 0.09}, | {"Missionaries of Temple Alabaster", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, | ||
{"Missionaries of the Court of the Ironclad", "Temple Bank of the Reformed Stripping Path", 0.06}, | {"Missionaries of the Court of the Ironclad", "Temple Bank of the Reformed Stripping Path", 0.06, "teams"}, | ||
{"Missionaries of Concord Accordia", "Temple Bank of the Reformed Stripping Path", 0.02}, | {"Missionaries of Concord Accordia", "Temple Bank of the Reformed Stripping Path", 0.02, "teams"}, | ||
{"Missionaries of the Mystery of the Stygian Veil", "Temple Bank of the Reformed Stripping Path", 0.03}, | {"Missionaries of the Mystery of the Stygian Veil", "Temple Bank of the Reformed Stripping Path", 0.03, "teams"}, | ||
{"Missionaries of the Sylvan Fellowship", "Temple Bank of the Reformed Stripping Path", 0.09}, | {"Missionaries of the Sylvan Fellowship", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, | ||
{"Missionaries of the Mystery of Red Mirth", "Temple Bank of the Reformed Stripping Path", 0.06}, | {"Missionaries of the Mystery of Red Mirth", "Temple Bank of the Reformed Stripping Path", 0.06, "teams"}, | ||
{"Missionaries of Rex Catonis", "Temple Bank of the Reformed Stripping Path", 0.07}, | {"Missionaries of Rex Catonis", "Temple Bank of the Reformed Stripping Path", 0.07, "teams"}, | ||
{"Missionaries of Sanctum Delphica", "Temple Bank of the Reformed Stripping Path", 0.09}, | {"Missionaries of Sanctum Delphica", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, | ||
{"Missionaries of Ordo Amictia", "Temple Bank of the Reformed Stripping Path", 0.06}, | {"Missionaries of Ordo Amictia", "Temple Bank of the Reformed Stripping Path", 0.06, "teams"}, | ||
{"Missionaries of Temple Illuminata", "Temple Bank of the Reformed Stripping Path", 0.05}, | {"Missionaries of Temple Illuminata", "Temple Bank of the Reformed Stripping Path", 0.05, "teams"}, | ||
{"Couriers", "Couriers of the Lizard Queen", 0.04}, | {"Couriers", "Couriers of the Lizard Queen", 0.04, "dispatches"}, | ||
{"Wisp Ward Talismans", "Couriers of the Lizard Queen", 0.10}, | {"Wisp Ward Talismans", "Couriers of the Lizard Queen", 0.10, "amulets"}, | ||
{"Credit Hours", "Temple University of Delphica", 3.00}, | {"Credit Hours", "Temple University of Delphica", 3.00, "credits"}, | ||
{"Circuses", "La Sái Ebile", 0.24}, | {"Circuses", "La Sái Ebile", 0.24, "venues"}, | ||
{"Performances", "La Sái Ebile", 0.11}, | {"Performances", "La Sái Ebile", 0.11, "acts"}, | ||
{"Hellhound (scout)", "Hellhound Breeders of Dragevik", 0.03}, | {"Hellhound (scout)", "Hellhound Breeders of Dragevik", 0.03, "hounds"}, | ||
{"Hellhound (soldier)", "Hellhound Breeders of Dragevik", 0.05}, | {"Hellhound (soldier)", "Hellhound Breeders of Dragevik", 0.05, "hounds"}, | ||
{"Hellhound (spy)", "Hellhound Breeders of Dragevik", 0.15}, | {"Hellhound (spy)", "Hellhound Breeders of Dragevik", 0.15, "hounds"}, | ||
} | } | ||
--------------------------------------------------------- | |||
-- Special Date Events | |||
--------------------------------------------------------- | |||
local events = { | local events = { | ||
["1,6"] = "Bayram al-Nur (Festival of Light) [Vaeringheim]", | ["1,6"] = "Bayram al-Nur (Festival of Light) [Vaeringheim]", | ||
["1,18"] = "Chag Or Hadash (Festival of New Light) [Luminaria]", | ["1,18"] = "Chag Or Hadash (Festival of New Light) [Luminaria]", | ||
["1,30"] = "Symposion Eirinis (Symposium of Harmony) [Serena]", | ["1,30"] = "Symposion Eirinis (Symposium of Harmony) [Serena]", | ||
Line 75: | Line 92: | ||
["1,48"] = "Day of Abandonment [Vaeringheim]", | ["1,48"] = "Day of Abandonment [Vaeringheim]", | ||
["1,55"] = "Tikkun Tzel (Repair of Shadows) [Symphonara]", | ["1,55"] = "Tikkun Tzel (Repair of Shadows) [Symphonara]", | ||
["2,6"] = "Panegyris Chrysou (Golden Gathering) [Aurelia]", | ["2,6"] = "Panegyris Chrysou (Golden Gathering) [Aurelia]", | ||
["2,17"] = "Constitution Day [Luminaria]", | ["2,17"] = "Constitution Day [Luminaria]", | ||
["2,19"] = "Mehtap Dalgası (Moonlit Tide) [Vaeringheim]", | ["2,19"] = "Mehtap Dalgası (Moonlit Tide) [Vaeringheim]", | ||
Line 85: | Line 102: | ||
["2,54"] = "Leilat al-Kamar (Night of the Moon) [Lunalis Sancta]", | ["2,54"] = "Leilat al-Kamar (Night of the Moon) [Lunalis Sancta]", | ||
["2,61"] = "Taşrakah (Reverence of the Stone) [Luminaria]", | ["2,61"] = "Taşrakah (Reverence of the Stone) [Luminaria]", | ||
["3,6"] = "Chag Tvuah (Festival of Harvest) [Sylvapolis]", | ["3,6"] = "Chag Tvuah (Festival of Harvest) [Sylvapolis]", | ||
["3,18"] = "Anagenesis Eirmos (Procession of Rebirth) [Acheron]", | ["3,18"] = "Anagenesis Eirmos (Procession of Rebirth) [Acheron]", | ||
["3,28"] = "Panagia Therizis (Holy Day of the Reaper) [Sylvapolis]", | ["3,28"] = "Panagia Therizis (Holy Day of the Reaper) [Sylvapolis]", | ||
["3,37"] = "Anniversary of victory in the Morovian Frontier Campaign [Acheron]", | ["3,37"] = "Anniversary of victory in the Morovian Frontier Campaign [Acheron]", | ||
["3,43"] = "Karnavali Thysias (Carnival of Celebration) [Erythros]", | ["3,43"] = "Karnavali Thysias (Carnival of Celebration) [Erythros]", | ||
["3,53"] = "Sefar Yashar (Straight Path Celebration) [Catonis Atrium]" | ["3,53"] = "Sefar Yashar (Straight Path Celebration) [Catonis Atrium]" | ||
} | } | ||
--------------------------------------------------------- | |||
-- simulatePriceChange | |||
-- Applies a daily random factor; special date events | |||
-- add a small positive boost to the final price. | |||
--------------------------------------------------------- | |||
local function simulatePriceChange(basePrice, dayOfMonth, month) | local function simulatePriceChange(basePrice, dayOfMonth, month) | ||
local dateKey = string.format("%d,%d", month, dayOfMonth) | local dateKey = string.format("%d,%d", month, dayOfMonth) | ||
local changeFactor = math.random(-20, 20) / 100 | local changeFactor = math.random(-20, 20) / 100 | ||
local explanation | local explanation | ||
if events[dateKey] then | if events[dateKey] then | ||
explanation = events[dateKey] .. ": Prices influenced by event dynamics." | explanation = events[dateKey] .. ": Prices influenced by event dynamics." | ||
changeFactor = changeFactor + 0.05 -- | changeFactor = changeFactor + 0.05 -- small event-based boost | ||
else | else | ||
explanation = "General market fluctuations." | explanation = "General market fluctuations." | ||
Line 109: | Line 131: | ||
end | end | ||
--------------------------------------------------------- | |||
-- determinePriceLevel | |||
-- If adjustedPrice < basePrice => "High", else "Low" | |||
--------------------------------------------------------- | |||
local function determinePriceLevel(basePrice, adjustedPrice) | |||
if adjustedPrice < basePrice then | |||
return "High" | |||
else | |||
return "Low" | |||
end | |||
end | |||
--------------------------------------------------------- | |||
-- Main function: generateTable | |||
-- Seeds random daily, then builds table | |||
--------------------------------------------------------- | |||
function p.generateTable() | function p.generateTable() | ||
local date, dayOfMonth, month = getCurrentDate() | -- (A) Create a daily seed based on current local date | ||
local wikitable = "{| class=\"wikitable\"\n! Date (Day/Month/Year [[PSSC]]) !! Product !! Producing Company !! Base Price !! Tier III Price !! Tier II Price !! Tier I Price !! Explanation\n" | local localTime = os.date("*t") | ||
local dailySeed = localTime.year * 1000 + localTime.yday | |||
math.randomseed(dailySeed) | |||
-- (B) Retrieve date for display and logic | |||
local dateString, dayOfMonth, month = getCurrentDate() | |||
-- (C) Table header | |||
local wikitable = | |||
"{| class=\"wikitable\"\n" .. | |||
"! Date (Day/Month/Year [[PSSC]]) !! Product !! Producing Company !! Base Price !! Price Level !! Tier III Price !! Tier II Price !! Tier I Price !! Units !! Explanation\n" | |||
-- We’ll track totals in order to compare average adjusted vs. average base | |||
local totalBase, totalNew = 0, 0 | |||
local count = 0 | |||
-- (D) Iterate over all products, randomize daily | |||
for _, product in ipairs(products) do | for _, product in ipairs(products) do | ||
local name | local name = product[1] | ||
local company = product[2] | |||
local basePrice = product[3] | |||
local units = product[4] | |||
local newPrice, changeFactor, explanation = simulatePriceChange(basePrice, dayOfMonth, month) | local newPrice, changeFactor, explanation = simulatePriceChange(basePrice, dayOfMonth, month) | ||
local priceLevel = determinePriceLevel(basePrice, newPrice) | |||
-- | |||
-- Tiers | |||
local tierIIIPrice = math.floor((newPrice / 3) * 100) / 100 | local tierIIIPrice = math.floor((newPrice / 3) * 100) / 100 | ||
local tierIIPrice = math.floor((newPrice * 2 / 3) * 100) / 100 | local tierIIPrice = math.floor((newPrice * 2 / 3) * 100) / 100 | ||
local tierIPrice = newPrice | local tierIPrice = newPrice -- final adjusted | ||
-- Accumulate totals for later | |||
totalBase = totalBase + basePrice | |||
totalNew = totalNew + newPrice | |||
count = count + 1 | |||
wikitable = wikitable .. string.format("|-\n| %s || %s || %s || %.2f || %.2f || %.2f || %.2f || %s\n", | -- Build row | ||
wikitable = wikitable .. string.format( | |||
"|-\n| %s || %s || %s || %.2f || %s || %.2f || %.2f || %.2f || %s || %s\n", | |||
dateString, name, company, basePrice, priceLevel, | |||
tierIIIPrice, tierIIPrice, tierIPrice, units, explanation | |||
) | |||
end | end | ||
-- (E) Compute average base vs. new | |||
local avgBase = totalBase / count | |||
local avgNew = totalNew / count | |||
-- (F) Determine overall price level using the same logic | |||
local averagePriceLevel | |||
if avgNew < avgBase then | |||
averagePriceLevel = "High" | |||
else | |||
averagePriceLevel = "Low" | |||
end | |||
-- (G) Add final row to show average price comparison | |||
-- Note the colspan="10" to match the number of columns | |||
wikitable = wikitable .. string.format( | |||
"|-\n! colspan=\"10\" | Average adjusted price (%.2f) vs. average base price (%.2f) => **%s**\n", | |||
avgNew, avgBase, averagePriceLevel | |||
) | |||
-- Close the table | |||
wikitable = wikitable .. "|}" | wikitable = wikitable .. "|}" | ||
return wikitable | return wikitable |
Latest revision as of 17:06, 2 January 2025
Documentation for this module may be created at Module:AutoUpdateServPrices/doc
--------------------------------------------------------- -- Module:DailyPriceTable -- Generates a daily-updated table where each cell -- refreshes once per day, with random fluctuations -- stable for that day but new on subsequent days. --------------------------------------------------------- local p = {} --------------------------------------------------------- -- getCurrentDate -- Returns today's date in PSSC format, -- along with dayOfMonth and month for daily logic. --------------------------------------------------------- local function getCurrentDate() local startDate = os.time({year = 1999, month = 8, day = 6}) local secondsInDay = 86400 local daysPerYear = 183 local currentDate = os.time() local totalDaysElapsed = math.floor((currentDate - startDate) / secondsInDay) local yearFraction = totalDaysElapsed / daysPerYear local psscYear = math.floor(yearFraction) local dayOfYear = math.floor((yearFraction - psscYear) * daysPerYear) + 1 local month, dayOfMonth if dayOfYear <= 61 then month = 1 dayOfMonth = dayOfYear elseif dayOfYear <= 122 then month = 2 dayOfMonth = dayOfYear - 61 else month = 3 dayOfMonth = dayOfYear - 122 end return string.format("%d/%d/%d [[PSSC]]", dayOfMonth, month, psscYear), dayOfMonth, month end --------------------------------------------------------- -- Product data remains the same --------------------------------------------------------- local products = { {"Currency", "Lake Morovia Blockade Fund", 4.05, "stacks"}, {"Plunder", "Lake Morovia Blockade Fund", 3.30, "chests"}, {"Prostitutes", "Court of the Dark Harpy", 0.15, "contracts"}, {"Lievs", "Hatch Ministry", 0.32, "units"}, {"Prisoners", "Hatch Ministry", 0.04, "cells"}, {"Faces", "Maritime Guild of the Cult of Maskmakers", 0.03, "masks"}, {"Spies", "Ergonian Spy Agency", 0.01, "assignments"}, {"Missionaries of the Order Aurora Mystica", "Temple Bank of the Reformed Stripping Path", 0.05, "teams"}, {"Missionaries of Harmony Sanctum", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, {"Missionaries of Ignis Aeternum", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, {"Missionaries of Celestial Harmony Sect", "Temple Bank of the Reformed Stripping Path", 0.08, "teams"}, {"Missionaries of the Guild of Golden Shadows", "Temple Bank of the Reformed Stripping Path", 0.05, "teams"}, {"Missionaries of the Azure Sentinel Sect", "Temple Bank of the Reformed Stripping Path", 0.01, "teams"}, {"Missionaries of Reverie Nebulous", "Temple Bank of the Reformed Stripping Path", 0.10, "teams"}, {"Missionaries of the Eon Fellowship", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, {"Missionaries of the Order of the Umbral Oracle", "Temple Bank of the Reformed Stripping Path", 0.07, "teams"}, {"Missionaries of the Mystery of the Verdant Embrace", "Temple Bank of the Reformed Stripping Path", 0.10, "teams"}, {"Missionaries of Conclace Illuminara", "Temple Bank of the Reformed Stripping Path", 0.02, "teams"}, {"Missionaries of Sanctum Vitalis", "Temple Bank of the Reformed Stripping Path", 0.06, "teams"}, {"Missionaries of Temple Alabaster", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, {"Missionaries of the Court of the Ironclad", "Temple Bank of the Reformed Stripping Path", 0.06, "teams"}, {"Missionaries of Concord Accordia", "Temple Bank of the Reformed Stripping Path", 0.02, "teams"}, {"Missionaries of the Mystery of the Stygian Veil", "Temple Bank of the Reformed Stripping Path", 0.03, "teams"}, {"Missionaries of the Sylvan Fellowship", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, {"Missionaries of the Mystery of Red Mirth", "Temple Bank of the Reformed Stripping Path", 0.06, "teams"}, {"Missionaries of Rex Catonis", "Temple Bank of the Reformed Stripping Path", 0.07, "teams"}, {"Missionaries of Sanctum Delphica", "Temple Bank of the Reformed Stripping Path", 0.09, "teams"}, {"Missionaries of Ordo Amictia", "Temple Bank of the Reformed Stripping Path", 0.06, "teams"}, {"Missionaries of Temple Illuminata", "Temple Bank of the Reformed Stripping Path", 0.05, "teams"}, {"Couriers", "Couriers of the Lizard Queen", 0.04, "dispatches"}, {"Wisp Ward Talismans", "Couriers of the Lizard Queen", 0.10, "amulets"}, {"Credit Hours", "Temple University of Delphica", 3.00, "credits"}, {"Circuses", "La Sái Ebile", 0.24, "venues"}, {"Performances", "La Sái Ebile", 0.11, "acts"}, {"Hellhound (scout)", "Hellhound Breeders of Dragevik", 0.03, "hounds"}, {"Hellhound (soldier)", "Hellhound Breeders of Dragevik", 0.05, "hounds"}, {"Hellhound (spy)", "Hellhound Breeders of Dragevik", 0.15, "hounds"}, } --------------------------------------------------------- -- Special Date Events --------------------------------------------------------- local events = { ["1,6"] = "Bayram al-Nur (Festival of Light) [Vaeringheim]", ["1,18"] = "Chag Or Hadash (Festival of New Light) [Luminaria]", ["1,30"] = "Symposion Eirinis (Symposium of Harmony) [Serena]", ["1,43"] = "Alev Günü (Day of Flame) [Pyralis]", ["1,48"] = "Day of Abandonment [Vaeringheim]", ["1,55"] = "Tikkun Tzel (Repair of Shadows) [Symphonara]", ["2,6"] = "Panegyris Chrysou (Golden Gathering) [Aurelia]", ["2,17"] = "Constitution Day [Luminaria]", ["2,19"] = "Mehtap Dalgası (Moonlit Tide) [Vaeringheim]", ["2,31"] = "Oneiro Foteino (Dream of Illumination) [Somniumpolis]", ["2,39"] = "Anniversary of victory in the New South Jangsong Campaign [Nexa]", ["2,44"] = "Erev Galgal (Eve of Cycles) [Nexa]", ["2,45"] = "Bassaridia Festival [Somniumpolis]", ["2,48"] = "Anniversary of victory in the Southern Lake Morovia Campaign [Somniumpolis]", ["2,54"] = "Leilat al-Kamar (Night of the Moon) [Lunalis Sancta]", ["2,61"] = "Taşrakah (Reverence of the Stone) [Luminaria]", ["3,6"] = "Chag Tvuah (Festival of Harvest) [Sylvapolis]", ["3,18"] = "Anagenesis Eirmos (Procession of Rebirth) [Acheron]", ["3,28"] = "Panagia Therizis (Holy Day of the Reaper) [Sylvapolis]", ["3,37"] = "Anniversary of victory in the Morovian Frontier Campaign [Acheron]", ["3,43"] = "Karnavali Thysias (Carnival of Celebration) [Erythros]", ["3,53"] = "Sefar Yashar (Straight Path Celebration) [Catonis Atrium]" } --------------------------------------------------------- -- simulatePriceChange -- Applies a daily random factor; special date events -- add a small positive boost to the final price. --------------------------------------------------------- local function simulatePriceChange(basePrice, dayOfMonth, month) local dateKey = string.format("%d,%d", month, dayOfMonth) local changeFactor = math.random(-20, 20) / 100 local explanation if events[dateKey] then explanation = events[dateKey] .. ": Prices influenced by event dynamics." changeFactor = changeFactor + 0.05 -- small event-based boost else explanation = "General market fluctuations." end local newPrice = math.floor((basePrice * (1 + changeFactor)) * 100) / 100 return newPrice, changeFactor, explanation end --------------------------------------------------------- -- determinePriceLevel -- If adjustedPrice < basePrice => "High", else "Low" --------------------------------------------------------- local function determinePriceLevel(basePrice, adjustedPrice) if adjustedPrice < basePrice then return "High" else return "Low" end end --------------------------------------------------------- -- Main function: generateTable -- Seeds random daily, then builds table --------------------------------------------------------- function p.generateTable() -- (A) Create a daily seed based on current local date local localTime = os.date("*t") local dailySeed = localTime.year * 1000 + localTime.yday math.randomseed(dailySeed) -- (B) Retrieve date for display and logic local dateString, dayOfMonth, month = getCurrentDate() -- (C) Table header local wikitable = "{| class=\"wikitable\"\n" .. "! Date (Day/Month/Year [[PSSC]]) !! Product !! Producing Company !! Base Price !! Price Level !! Tier III Price !! Tier II Price !! Tier I Price !! Units !! Explanation\n" -- We’ll track totals in order to compare average adjusted vs. average base local totalBase, totalNew = 0, 0 local count = 0 -- (D) Iterate over all products, randomize daily for _, product in ipairs(products) do local name = product[1] local company = product[2] local basePrice = product[3] local units = product[4] local newPrice, changeFactor, explanation = simulatePriceChange(basePrice, dayOfMonth, month) local priceLevel = determinePriceLevel(basePrice, newPrice) -- Tiers local tierIIIPrice = math.floor((newPrice / 3) * 100) / 100 local tierIIPrice = math.floor((newPrice * 2 / 3) * 100) / 100 local tierIPrice = newPrice -- final adjusted -- Accumulate totals for later totalBase = totalBase + basePrice totalNew = totalNew + newPrice count = count + 1 -- Build row wikitable = wikitable .. string.format( "|-\n| %s || %s || %s || %.2f || %s || %.2f || %.2f || %.2f || %s || %s\n", dateString, name, company, basePrice, priceLevel, tierIIIPrice, tierIIPrice, tierIPrice, units, explanation ) end -- (E) Compute average base vs. new local avgBase = totalBase / count local avgNew = totalNew / count -- (F) Determine overall price level using the same logic local averagePriceLevel if avgNew < avgBase then averagePriceLevel = "High" else averagePriceLevel = "Low" end -- (G) Add final row to show average price comparison -- Note the colspan="10" to match the number of columns wikitable = wikitable .. string.format( "|-\n! colspan=\"10\" | Average adjusted price (%.2f) vs. average base price (%.2f) => **%s**\n", avgNew, avgBase, averagePriceLevel ) -- Close the table wikitable = wikitable .. "|}" return wikitable end return p