
Better Veinminer
A Paper 1.21 veinminer plugin that mines entire ore veins instantly. Uses strict same-type BFS matching, so adjacent ores are never accidentally destroyed. Fully configurable via config.yml.
Better Veinminer 1.4.0
release14 мая 2026 г.[1.4.0] — 2026-05-14
Fixed
-
[CRITICAL] Async save race condition —
StatsManager.saveAll()was scheduled as an async task and readPlayerStatsfields directly while the main thread was still writing to them. A player could lose stats or receive a corrupted save file depending on thread timing. The auto-save timer now runs on the main thread to take a point-in-time snapshot via the newPlayerStats.copy(), then hands each snapshot to a single-threadedExecutorServicefor file I/O. The snapshot is fully consistent; the I/O thread never sees a half-written state. -
[CRITICAL] Concurrent YAML write corruption — The async auto-save timer and the
onQuitasync save could both be writing the same player's file simultaneously, producing a truncated or structurally invalid YAML file. All disk writes are now submitted to a single-threadedExecutorService(BVM-StatsIO). Only one write per file can be in progress at any time.StatsManager.shutdown()called fromonDisable()blocks up to 5 seconds to guarantee all pending writes complete before the plugin unloads. -
[CRITICAL] Daily reset based on server uptime, not real calendar date — The reset used
24 * 60 * 60 * 20Lticks (24 hours of server uptime). TPS lag caused the timer to drift, and a server restart before 24 hours elapsed reset the timer entirely, effectively disabling the daily limit. Replaced with aLocalDate.now()comparison: the last reset date is persisted toplugins/BetterVeinminer/data.ymland checked on startup (to catch any resets missed while the server was offline) and every 2 minutes (to detect midnight without requiring an exact timer). -
[HIGH] Cooldown cleanup removed entries too aggressively — The cleanup task that evicts stale entries from the
cooldownsmap used a hardcoded 60-second cutoff (System.currentTimeMillis() - 60_000L). If the configured cooldown exceeded 60 seconds, player entries were evicted while the cooldown was still active. On the next vein break the lookup returnednulland the cooldown was silently bypassed. The cutoff now usesconfig.getCooldownMs(). -
[HIGH] Protection plugin bypass — Extra blocks in the vein were broken with
block.breakNaturally(tool)without first firing aBlockBreakEvent. WorldGuard, GriefPrevention, Lands, and Residence listen toBlockBreakEventto enforce region protection; because no event was fired, blocks in protected regions could be destroyed. ABlockBreakEventis now fired for each extra block before it is broken; if any listener cancels the event, that block is skipped. -
[HIGH] Missing anti-recursion guard — The protection-plugin fix above fires
BlockBreakEventprogrammatically, which caused the plugin's ownonBlockBreaklistener to trigger again for each block, scheduling a newVeinmineTaskper block and creating an infinite chain. AminingPlayersSet<UUID>marks any player whose veinmine session is currently active;onBlockBreakreturns immediately if the player's UUID is already present. The set is always cleared in afinallyblock to ensure a player can never be permanently locked out. -
[MEDIUM] 26-direction BFS linked separate nearby veins — The BFS scanned all 26 neighbours (3×3×3 cube), allowing diagonal connections to bridge two distinct ore veins that happened to be close together. Vanilla Minecraft generates ores using face-adjacent placement only, so diagonal connections are always false positives. Changed to 6-direction (face-adjacent) BFS. This also reduces the maximum queue size by up to 4×, improving TPS on large veins.
-
[MEDIUM] Unbreaking durability formula was deterministic, not vanilla-accurate — The old
calcDamage()computedMath.round(hits × multiplier × 1/(unbreaking+1))and clamped to a minimum of 1. Vanilla Minecraft independently rolls each hit: each has a1/(unbreaking+1)chance to actually apply damage. The clamped average could never produce 0 damage even with high Unbreaking levels and few hits.calcDamage()now performs a per-hit Bernoulli trial usingMath.random(), matching vanilla behaviour including the possibility of 0 damage. -
[MEDIUM]
PlayerStatsexposed public mutable fields — All four fields (blocksMined,totalOresCollected,currentTier,veinminesUsed) werepublic, allowing any code to set them to negative or overflowed values without validation. All fields are nowprivate. Setters clamp values to valid ranges (e.g. tier ≥ 1, counts ≥ 0). The newcopy()method supports safe async snapshotting. -
[MEDIUM] Config accepted invalid numeric values — Values such as
max-blocks: -1ordamage-multiplier: -100were loaded without validation, producing undefined behaviour (negative BFS limit, inverted durability, server lag). All numeric fields are now clamped on load:max-blocks→[1, 1000],cooldown-ms→[0, 300000],damage-multiplier→[0.1, 10.0],limit-per-day→[1, 100000],base-exp→[0, 10000].
Clarified (not a bug)
- Tool swap exploit (audit report #7) — The report described a tool-swap exploit
where a player could switch items during the 1-tick task delay to bypass restrictions.
This was already fixed in v1.2.1:
VeinmineTaskrecords the inventory slot and tool snapshot at event time, then re-validates withisSimilar()before applying durability. No change needed.
Changed
- Auto-save timer switched from
runTaskTimerAsynchronouslytorunTaskTimer(main thread) so snapshot collection is always synchronised with game state. The actual file I/O is still off the main thread via theBVM-StatsIOexecutor. onDisable()now callsstatsManager.shutdown()(blocking flush with 5 s timeout) instead ofsaveAll()directly, guaranteeing all pending I/O completes before the plugin unloads.
Better Veinminer 1.3.0
release25 апреля 2026 г.[1.3.0] — 2026-04-25
Added
BetterVeinmineEvent— Custom cancellable Bukkit event fired just before extra blocks are broken. Other plugins can now:- Cancel the entire veinmine via
event.setCancelled(true) - Inspect the block list via
event.getBlocks()(unmodifiable) - Override the EXP reward via
event.setExpReward(int) - Read the ore type via
event.getOreType()
- Cancel the entire veinmine via
StatsManager— Full YAML persistence forPlayerStats. Stats are now saved toplugins/BetterVeinminer/stats/<uuid>.ymland survive server restarts.- Stats load synchronously on
PlayerJoinEvent(file is tiny, no noticeable delay) - Stats save asynchronously on
PlayerQuitEvent - Auto-save every 5 minutes (async) while the server is running
- Final synchronous flush in
onDisable()to guarantee no data loss on shutdown
- Stats load synchronously on
Fixed
- NullPointerException on missing config sections —
PluginConfig.reload()called.getKeys(false)directly on the return value ofgetConfigurationSection(...), which returnsnullwhen the section is absent fromconfig.yml. This caused an NPE crash on any server whereore-multipliers.multipliers,per-ore-cooldowns.cooldowns, orpermission-levelswere omitted or empty. All three sites now null-check the section before iterating. - Player-offline NPE in
VeinmineTask— The task is scheduled 1 tick after theBlockBreakEvent. If the player disconnects in that window (network drop, kick, crash), every subsequent field access on thePlayerobject threw aNullPointerException. Added an earlyif (!player.isOnline()) return;guard at the top ofrun(). {uses}placeholder never replaced in/bvm stats—showStats()built the message with{blocks}and{tier}substitutions but forgot{uses}, so the literal string{uses}was printed to the player.{blocks}placeholder never replaced in tier-up message — The tier-up message template"You reached Tier {tier}! New max blocks: {blocks}"only had{tier}replaced;{blocks}was left as-is in the output.sendActionBar(String)deprecated in Paper 1.21 — Replaced with the Adventure API:player.sendActionBar(LegacyComponentSerializer.legacySection().deserialize(msg)). This eliminates the deprecation warning in server logs and future-proofs the call.- Scheduler tasks not cancelled on disable —
onDisable()cleared data maps but never calledBukkit.getScheduler().cancelTasks(this). The cooldown-cleanup and daily-reset runnables could fire after the plugin was disabled (e.g. during/reload), accessing a partially torn-down plugin instance.cancelTasks(this)is now the first action inonDisable(), before any data is flushed or cleared.
Changed
onDisable()now cancels all plugin tasks before flushing data, then callsstatsManager.saveAll()as a final synchronous write to guarantee persistence.VeinmineTasknow collects extra blocks into aList<Block>(instead of iteratingSet<Block>with an origin equality check) to pass cleanly toBetterVeinmineEvent.- EXP calculation is now performed before the event is fired so listeners can read
and override the reward via
event.setExpReward(int).
Better Veinminer 1.2.2
release27 марта 2026 г.fix bug
Better Veinminer 1.2.1
release16 марта 2026 г.[1.2.1] — 2026-03-16
Fixed
- Critical Item Swap Bug — Fixed a bug where switching items during a veinmine task (0.05s delay) could overwrite a new item in the player's hand with the old tool, potentially causing item loss. The task now tracks the specific inventory slot and verifies the tool's identity before applying durability changes.
- UX Improvement — Success messages are now sent to the Action Bar instead of the chat to prevent spamming the chat history for active miners.
Better Veinminer 1.2.0
release14 марта 2026 г.[1.2.0] — 2026-03-14
Added
- Tier System — Players automatically progress through 4 tiers as they mine more blocks:
- Tier 1: 32 max blocks (0 blocks mined)
- Tier 2: 64 max blocks (100k blocks mined)
- Tier 3: 128 max blocks (500k blocks mined)
- Tier 4: 256 max blocks (1M blocks mined)
- Player Statistics — Track blocks mined, veinmines used, and current tier
- New Commands:
/bvm statsand/bvm tierfor player progression tracking - Ore Multipliers — Configure different drop multipliers for different ore types (e.g., diamonds worth 1.5x)
- Per-Ore Cooldowns — Optional: Set different cooldown times per ore type
- Permission-based Limits — VIP (
betterveinminer.vip) and Premium (betterveinminer.premium) tiers - Daily Limits — Optional: Prevent abuse by limiting veinmines per player per day
- Advanced Effect Settings — Customizable particle types, colors, speeds, and sounds
- Custom Messages — Fully customizable player notifications with placeholders
- EXP Rewards — Optional custom experience points for veinmining
- PlayerStats.java — New data class for tracking individual player progress
Changed
- Version bumped to 1.2.0 with major feature expansion
- pom.xml — Fixed XML syntax errors (typo:
<n>→<n>) - PluginConfig.java — Expanded from ~90 lines to ~300+ lines with comprehensive settings
- BetterVeinminer.java — Refactored to support tier system, stats, permissions, and daily limits
- config.yml — Completely rewritten with organized sections for each feature
- plugin.yml — Added 3 new commands and 2 new permission levels
Improved
- Much better tier progression system that rewards dedicated miners
- More granular permission-based control for VIP/Premium players
- Enhanced logging and player feedback with customizable messages
- Better code organization with dedicated helper methods
