FLINT AND STEEL!!!! THE NETHER!!!!!! RELEASE!!!!!!!!!

* Account bio support (fetch only route right now)
* Room cloning fixes
    - Dorm Room cloning is still broken
* Instance changing fixes
* Presence: VRMovementMode and StatusVisibility updates automatically
* Routes for the above two properties
* Settings can take numbers, too (enums)
* No microtransations in my game (parental controls)
* A whole lotta routes for various unfinished but planned features
    - Equipment
    - Consumables
    - Objectives
    - Checklist (orientation rewards)
    - Objectives (three daily tasks)
    - Image metadata
    - Community Board
    - Player Events
    - Storefronts
* Matchmaking instance querying
    - Empty instances are not yet cleared
* Avatar items, saved avatars, save current avatar routes
* No loading screen tips for now
* Send presence at an interval over the socket
    - Error FROSTBITE is reported in the game logs during bootup sometimes. Maybe due to the lack of ping messages?
* Socket push notifications

Note to self: Set up deno compilation in runners on gitea
This commit is contained in:
2025-04-02 23:56:18 -04:00
parent bcee414004
commit 1cfd0426dd
35 changed files with 758 additions and 64 deletions

View File

@@ -131,7 +131,7 @@ class RoomsBase {
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.Name, details.Room.Name),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.Description, details.Room.Description),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.CreatorPlayerId, details.Room.CreatorPlayerId),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.ImageName, `DefaultProfileImage.png`),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.ImageName, details.Room.ImageName),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.State, details.Room.State),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.Accessibility, details.Room.Accessibility),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.SupportsLevelVoting, `${details.Room.SupportsLevelVoting}`),
@@ -144,7 +144,7 @@ class RoomsBase {
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.AllowsJuniors, `${details.Room.AllowsJuniors}`),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.RoomWarningMask, details.Room.RoomWarningMask),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.CustomRoomWarning, details.Room.CustomRoomWarning),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.DisableMicAutoMute, `${details.Room.DisableMicAutoMute}`),
Redis.Database.hset(roomMetaRootKey, this.roomMetaKeys.DisableMicAutoMute, `${details.Room.DisableMicAutoMute ? details.Room.DisableMicAutoMute : 'null'}`),
]);
for (const subroom of details.Scenes) {
@@ -159,7 +159,7 @@ class RoomsBase {
await Promise.all([
Redis.Database.hset(subRootMetaKey, this.subroomMetaKeys.RoomSceneLocationId, subroom.RoomSceneLocationId),
Redis.Database.hset(subRootMetaKey, this.subroomMetaKeys.Name, subroom.RoomSceneLocationId),
Redis.Database.hset(subRootMetaKey, this.subroomMetaKeys.Name, subroom.Name),
Redis.Database.hset(subRootMetaKey, this.subroomMetaKeys.IsSandbox, `${subroom.IsSandbox}`),
Redis.Database.hset(subRootMetaKey, this.subroomMetaKeys.DataBlobName, subroom.DataBlobName),
Redis.Database.hset(subRootMetaKey, this.subroomMetaKeys.MaxPlayers, subroom.MaxPlayers),
@@ -168,9 +168,9 @@ class RoomsBase {
]);
await Redis.Database.sadd(Redis.buildKey(Redis.KeyGroups.Rooms.Root, details.Room.RoomId.toString(), this.roomRootKeys.Subrooms), newSubId);
}
}
await Redis.Database.set(Redis.buildKey(Redis.KeyGroups.Room_Names, details.Room.Name), details.Room.RoomId);
}
async cloneRoom(roomid: number, newname: string, newowner: Profile, dorm?: boolean) {
const canBeClonedRaw = await Redis.Database.hget(Redis.buildKey(
@@ -188,7 +188,7 @@ class RoomsBase {
if (!canBeCloned) return null;
const beforeRoom = await Rooms.get(roomid); // room must exist
if (!beforeRoom || !beforeRoom.Room.CloningAllowed) return null; // room must be cloneable
if (await Rooms.getByName(newname)) return null; // room name cannot be taken
if (await Rooms.getByName(newname) && beforeRoom.Room.Name !== 'DormRoom') return null; // room name cannot be taken
const newId = await this.#getAvailableRoomId();
beforeRoom.Room.CreatorPlayerId = newowner.getId();
@@ -197,9 +197,9 @@ class RoomsBase {
if (dorm) {
beforeRoom.Room.IsAGRoom = true;
beforeRoom.Room.IsDormRoom = true;
beforeRoom.Room.CloningAllowed = false;
}
await Redis.Database.set(Redis.buildKey(Redis.KeyGroups.Room_Names, newname), newId);
await Rooms.#setRoom(beforeRoom);
return beforeRoom;
@@ -217,17 +217,17 @@ class RoomsBase {
const baseDorm = await Rooms.getByName("DormRoom");
log.d('got base dorm');
log.d('Base dorm existed');
if (!baseDorm) return null;
log.d('base dorm is not null');
log.d('Base dorm was not null');
const newDorm = await this.cloneRoom(baseDorm.Room.RoomId, "DormRoom", player, true);
log.d('New dorm existed');
if (!newDorm) return null;
log.d('New dorm was not null');
await Redis.Database.hset(Redis.buildKey(
Redis.KeyGroups.Rooms.Root,
Redis.KeyGroups.Rooms.PlayerDorms
), player.getId().toString(), baseDorm.Room.RoomId);
log.d('got new dorm');
if (!newDorm) return null;
log.d('new dorm is not null');
), player.getId().toString(), newDorm.Room.RoomId);
return newDorm;
}
@@ -249,7 +249,7 @@ class RoomsBase {
SupportsLevelVoting: builtinRoom.SupportsLevelVoting,
IsAGRoom: true,
IsDormRoom: builtinRoom.Name == 'DormRoom',
CloningAllowed: builtinRoom.CloningAllowed,
CloningAllowed: builtinRoom.Name == 'DormRoom' ? true : builtinRoom.CloningAllowed,
SupportsScreens: builtinRoom.SupportsScreens,
SupportsWalkVR: builtinRoom.SupportsWalkVR,
SupportsTeleportVR: builtinRoom.SupportsTeleportVR,

View File

@@ -0,0 +1,31 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
export enum StorefrontTypes {
None,
LaserTag,
RecCenter,
Watch,
Quest_LostSkulls = 100,
Quest_Dracula,
RecRoyale = 200,
Cafe = 300,
Paintball = 400,
Bowling = 500,
StuntRunner = 600,
DormMirror = 700
}

View File

@@ -53,7 +53,7 @@ class MatchmakingBase {
loginLocks.delete(prof.getId());
}
async matchmake(options: MatchmakingOptions) {
async matchmake(options: MatchmakingOptions): Promise<MatchmakingResponse> {
if (options.instanceId) {
@@ -65,7 +65,7 @@ class MatchmakingBase {
if (instance.isFull) return { errorCode: MatchmakingErrorCode.InsufficientSpace }
else if (instance.isPrivate) return { errorCode: MatchmakingErrorCode.RoomInstanceIsPrivate };
Instances.setPlayerInstance(options.profile, instance);
await Instances.setPlayerInstance(options.profile, instance);
return { errorCode: MatchmakingErrorCode.Success, roomInstance: instance };
}
@@ -96,7 +96,7 @@ class MatchmakingBase {
} else {
Instances.setPlayerInstance(options.profile, foundInstance);
await Instances.setPlayerInstance(options.profile, foundInstance);
return { errorCode: MatchmakingErrorCode.Success, roomInstance: foundInstance };
}

View File

@@ -19,6 +19,7 @@ import Logging from "@proxnet/undead-logging";
import { Profile } from "../profiles.ts";
import { RoomInstance, InstanceOptions } from "./types.ts";
import { Config } from "../../config.ts";
import Presence from "./presence.ts";
const log = new Logging("Instances");
@@ -150,7 +151,7 @@ class InstancesBase {
*
* Synchronizes profile instance to `instance` and adds player to instance.
*/
setPlayerInstance(player: Profile, instance: RoomInstance) {
async setPlayerInstance(player: Profile, instance: RoomInstance) {
const currentInstance = player.getInstance();
if (currentInstance === instance) return;
@@ -161,7 +162,11 @@ class InstancesBase {
if (this.instanceCanFitPlayer(instance)) {
this.getInstancePlayers(instance).add(player);
player.setInstance(instance);
const pres = await Presence.get(player);
pres.update();
this.updateSingleInstanceIsFull(instance);
} else log.w(`Instance ${instance.roomInstanceId} is full. Cannot add player ${player.getId()}`);
}

View File

@@ -62,31 +62,43 @@ class PlayerPresence {
playerId: number;
statusVisibility: PlayerStatusVisibility;
deviceClass: DeviceClass;
vrMovementMode: VRMovementMode;
vrMovementMode: VRMovementMode | undefined;
roomInstance: RoomInstance | null;
lastSeen: Date;
async updateStatusVisibility() {
const PlayerStatusVisibilityEnum = z.nativeEnum(PlayerStatusVisibility);
type PlayerStatusVisibilityEnum = z.infer<typeof PlayerStatusVisibilityEnum>;
const visibilityResult = PlayerStatusVisibilityEnum.safeParse(await this.#profile.Settings.getSetting(SettingKey.PlayerStatusVisibility));
if (visibilityResult.success) this.statusVisibility = visibilityResult.data;
}
async updateVRMovementMode() {
const VRMovementModeEnum = z.nativeEnum(VRMovementMode);
const modeResult = VRMovementModeEnum.safeParse(await this.#profile.Settings.getSetting(SettingKey.VRMovementMode));
if (modeResult.success) this.vrMovementMode = modeResult.data;
}
async update() {
this.updateOffline();
if (!this.offline) await this.updateStatusVisibility();
else this.statusVisibility = PlayerStatusVisibility.Offline;
if (!this.offline) {
await this.updateStatusVisibility();
await this.updateVRMovementMode();
}
else {
this.vrMovementMode = undefined;
this.statusVisibility = PlayerStatusVisibility.Offline;
}
this.deviceClass = await this.#profile.getKnownDeviceClass();
this.vrMovementMode = await this.#profile.getVRMovementMode();
this.roomInstance = this.#profile.getInstance();
}
/**
* Export presence object. Please make sure to update values with `Presence.update()` (async) before calling this.
* Export presence
*/
async export() {
await this.update();
@@ -97,7 +109,7 @@ class PlayerPresence {
deviceClass: this.deviceClass,
vrMovementMode: this.vrMovementMode
}
return Object.assign({}, exp); // hard clone
return exp;
}
updateOffline() {
@@ -118,10 +130,8 @@ class PresenceBase {
*/
async getAllPresences() {
const presSet: Set<PresenceExport> = new Set();
for (const pres of presence.values()) {
await pres.update();
for (const pres of presence.values())
presSet.add(await pres.export());
}
return presSet;
}

View File

@@ -15,7 +15,7 @@ GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { IntegratedRoomScene, RoomDetails } from "../content/roomtypes.ts";
import { RoomDetails } from "../content/roomtypes.ts";
import { Profile } from "../profiles.ts";
export enum PhotonRegionCodeString {

View File

@@ -18,7 +18,7 @@ along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { Redis } from "../../db.ts";
import { ProfileContentManager } from "./base/profilemanagerbase.ts";
interface AvatarSettings {
export interface AvatarSettings {
OutfitSelections: string,
HairColor: string,
SkinColor: string,

View File

@@ -37,11 +37,11 @@ export class ProfileSettingsManager extends ProfileContentManager {
return await Redis.Database.hget(Redis.buildKey(Redis.KeyGroups.Profiles.Root, this.profileId.toString(), Redis.KeyGroups.Profiles.Settings), key);
}
async setSetting(key: SettingKey, value: string) {
async setSetting(key: SettingKey, value: string | number) {
await Redis.Database.hset(Redis.buildKey(Redis.KeyGroups.Profiles.Root, this.profileId.toString(), Redis.KeyGroups.Profiles.Settings), key, value);
}
async setSettingRaw(key: string, value: string) {
async setSettingRaw(key: string, value: string | number) {
await Redis.Database.hset(Redis.buildKey(Redis.KeyGroups.Profiles.Root, this.profileId.toString(), Redis.KeyGroups.Profiles.Settings), key, value);
}

View File

@@ -217,6 +217,16 @@ class Profile {
return (await Redis.Database.sismember(Redis.buildKey(Redis.KeyGroups.Operators), this.#id.toString())) >= 1;
}
async getBio() {
const bio = await Redis.Database.get(Redis.buildKey(Redis.KeyGroups.Profiles.Root, this.#id.toString(), Redis.KeyGroups.Profiles.Bio));
if (!bio) return "This player has not yet set their bio. Remind them to set one!";
else return bio;
}
async setBio(bio: string) {
await Redis.Database.set(Redis.buildKey(Redis.KeyGroups.Profiles.Root, this.#id.toString(), Redis.KeyGroups.Profiles.Bio), bio);
}
async export() {
return await Profile.getExportAccount(this.#id);
}
@@ -256,7 +266,7 @@ class Profile {
const data = await this.Settings.getSetting(SettingKey.VRMovementMode);
if (data == null) {
log.w(`No known VR movement mode for ${this.#id} (harmless if OOBE not ran)`);
log.w(`No known VR movement mode for ${this.#id}`);
return VRMovementMode.Teleport;
}
const parsedData = parseInt(data);

View File

@@ -86,6 +86,7 @@ export const KeyGroups = {
Settings: "settings",
DeviceClass: "deviceClass",
Xp: "xp",
Bio: "bio",
Relationships: {
Root: "relationships",
IncomingFriendRequests: "incomingFriendRequests",

View File

@@ -17,7 +17,9 @@ along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../apiutils.ts";
import { route as AccountRoute } from "./account/account.ts";
import { route as ParentalControlRoute } from "./account/parentalcontrol.ts";
export const route = APIUtils.createRouter("/accountservice");
route.router.use(AccountRoute.path, AccountRoute.router);
route.router.use(ParentalControlRoute.path, ParentalControlRoute.router);

View File

@@ -17,17 +17,11 @@ along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import express from "express";
import { Profile } from "../../data/profiles.ts";
import UnifiedProfile, { Profile } from "../../data/profiles.ts";
import { z } from "zod";
export const route = APIUtils.createRouter("/account");
interface CreateAccountRequestBody {
platform: string;
platformId: string;
deviceId: string;
}
const CreateAccountRequestBodySchema = z.object({
platform: z.string(),
platformId: z.string(),
@@ -106,4 +100,33 @@ route.router.get("/me",
},
);
interface BioFetchParams {
id?: string
}
route.router.get('/:id/bio',
APIUtils.Authentication,
async (rq: express.Request<BioFetchParams>, rs: express.Response) => {
const unparsedId = rq.params.id;
if (!unparsedId) {
rs.sendStatus(500);
return;
}
const parsedId = parseInt(unparsedId);
if (isNaN(parsedId)) {
rs.sendStatus(400);
return;
}
const player = UnifiedProfile.get(parsedId);
rs.json({
accountId: parsedId,
bio: await player.getBio(),
});
}
);

View File

@@ -0,0 +1,35 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import { AuthType } from "../../data/users.ts";
export const route = APIUtils.createRouter('/parentalcontrol');
route.router.get('/me',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
(_rq, rs) => {
rs.json({
accountId: rs.locals.profile.getId(),
disallowInAppPurchases: true // no transations here buddy
});
},
);

View File

@@ -30,6 +30,14 @@ import { route as AvatarRoute } from "./api/avatar.ts";
import { route as QuickPlayRoute } from "./api/quickplay.ts";
import { route as RoomsRoute } from "./api/rooms.ts";
import { route as ChallengeRoute } from "./api/challenge.ts";
import { route as EquipmentRoute } from "./api/equipment.ts";
import { route as ConsumablesRoute } from "./api/consumables.ts";
import { route as ObjectivesRoute } from "./api/objectives.ts";
import { route as ChecklistRoute } from "./api/checklist.ts";
import { route as ImagesRoute } from "./api/images.ts";
import { route as CommunityBoardRoute } from "./api/communityboard.ts";
import { route as PlayerEventsRoute } from "./api/playerevents.ts";
import { route as StorefrontsRoute } from "./api/storefronts.ts";
export const route = APIUtils.createRouter("/api");
@@ -46,4 +54,12 @@ route.router.use(PlayerReputationRoute.path, PlayerReputationRoute.router);
route.router.use(AvatarRoute.path, AvatarRoute.router);
route.router.use(QuickPlayRoute.path, QuickPlayRoute.router);
route.router.use(RoomsRoute.path, RoomsRoute.router);
route.router.use(ChallengeRoute.path, ChallengeRoute.router);
route.router.use(ChallengeRoute.path, ChallengeRoute.router);
route.router.use(EquipmentRoute.path, EquipmentRoute.router);
route.router.use(ConsumablesRoute.path, ConsumablesRoute.router);
route.router.use(ObjectivesRoute.path, ObjectivesRoute.router);
route.router.use(ChecklistRoute.path, ChecklistRoute.router);
route.router.use(ImagesRoute.path, ImagesRoute.router);
route.router.use(CommunityBoardRoute.path, CommunityBoardRoute.router);
route.router.use(PlayerEventsRoute.path, PlayerEventsRoute.router);
route.router.use(StorefrontsRoute.path, StorefrontsRoute.router);

View File

@@ -15,8 +15,11 @@ GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import { z } from "zod";
import { APIUtils, NoBody } from "../../apiutils.ts";
import { AuthType } from "../../data/users.ts";
import express from "express";
import { AvatarSettings } from "../../data/profile/avatar.ts";
export const route = APIUtils.createRouter("/avatar");
@@ -31,13 +34,49 @@ route.router.get('/v2',
);
const AvatarSettingsSchema = z.object({
OutfitSelections: z.string(),
HairColor: z.string(),
SkinColor: z.string(),
FaceFeatures: z.string()
});
route.router.post('/v2/set',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
express.json(),
APIUtils.validateRequestBody(AvatarSettingsSchema),
(rq: express.Request<NoBody, NoBody, AvatarSettings>, rs: express.Response) => {
rs.locals.profile.Avatar.setAvatar(rq.body);
rs.sendStatus(200);
},
);
route.router.get('/v4/items',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
(_rq, rs) => {
rs.json([]);
}
APIUtils.emptyArrayResponse
);
route.router.get('/v3/saved',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
APIUtils.emptyArrayResponse
);
route.router.get('/v2/gifts',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
APIUtils.emptyArrayResponse
);

View File

@@ -0,0 +1,30 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import { AuthType } from "../../data/users.ts";
export const route = APIUtils.createRouter('/checklist');
route.router.get('/v1/current',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
APIUtils.emptyArrayResponse
);

View File

@@ -0,0 +1,51 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import { Config } from "../../config.ts";
import { AuthType } from "../../data/users.ts";
const config = Config.getConfig();
export const route = APIUtils.createRouter('/communityboard');
route.router.get('/v1/current',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
(_rq, rs) => {
rs.json({
FeaturedPlayer: {
Id: 1,
TitleOverride: "",
UrlOverride: ""
},
FeaturedRoomGroup: {
Name: "",
FeaturedRooms: []
},
CurrentAnnouncement: {
Message: config.public.motd,
MoreInfoUrl: ""
},
InstagramImages: [],
Videos: []
});
},
);

View File

@@ -0,0 +1,30 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import { AuthType } from "../../data/users.ts";
export const route = APIUtils.createRouter('/consumables');
route.router.get('/v1/getUnlocked',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
APIUtils.emptyArrayResponse
);

View File

@@ -0,0 +1,30 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import { AuthType } from "../../data/users.ts";
export const route = APIUtils.createRouter('/equipment');
route.router.get('/v2/getUnlocked',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
APIUtils.emptyArrayResponse
)

View File

@@ -19,6 +19,4 @@ import { APIUtils } from "../../apiutils.ts";
export const route = APIUtils.createRouter("/gameconfigs");
route.router.get("/v1/all", (_rq, rs) => {
rs.json([]);
});
route.router.get("/v1/all", APIUtils.emptyArrayResponse);

30
src/routes/api/images.ts Normal file
View File

@@ -0,0 +1,30 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import { AuthType } from "../../data/users.ts";
export const route = APIUtils.createRouter('/images');
route.router.get('/v2/named',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
APIUtils.emptyArrayResponse
);

View File

@@ -23,8 +23,6 @@ route.router.get('/v2/get',
APIUtils.Authentication,
(_rq, rs) => {
rs.json([]); // temporary
}
APIUtils.emptyArrayResponse
)

View File

@@ -0,0 +1,35 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import { AuthType } from "../../data/users.ts";
export const route = APIUtils.createRouter('/objectives');
route.router.get('/v1/myprogress',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
(_rq, rs) => {
rs.json({
Objectives: [],
ObjectiveGroups: []
});
},
);

View File

@@ -0,0 +1,44 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import { AuthType } from "../../data/users.ts";
export const route = APIUtils.createRouter('/playerevents');
route.router.get('/v1/all',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
(_rq, rs) => {
rs.json({
Created: [],
Responses: []
});
},
);
route.router.get('/v1',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
APIUtils.emptyArrayResponse
);

View File

@@ -21,8 +21,6 @@ export const route = APIUtils.createRouter("/playersubscriptions");
route.router.get('/v1/my',
(_rq, rs) => {
rs.json([]); // temporary: todo
}
APIUtils.emptyArrayResponse
);

View File

@@ -17,6 +17,7 @@ along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import Rooms from "../../data/content/rooms.ts";
import { RoomAccessibility } from "../../data/content/roomtypes.ts";
import { AuthType } from "../../data/users.ts";
import express from "express";
@@ -46,4 +47,75 @@ route.router.get('/v4/details/:roomId',
else rs.json(room);
},
)
);
route.router.get('/v2/myrooms',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
APIUtils.emptyArrayResponse
);
route.router.get('/v1/hot',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
async (_rq, rs) => {
// temporary: return all public AG rooms for testing
const rooms = await Rooms.getAllBuiltinRoomGenerations();
rs.json(rooms.map(room => room.Room).filter(room => room.Accessibility !== RoomAccessibility.Private));
},
);
route.router.get('/v2/baserooms',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
async (_rq, rs) => {
const rooms = await Rooms.getAllBuiltinRoomGenerations();
rs.json(rooms.map(room => room.Room).filter(room => room.CloningAllowed));
},
);
interface GetRoomByNameParams {
name?: string
}
route.router.get('/v2/name/:name',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
async (rq: express.Request<GetRoomByNameParams>, rs: express.Response) => {
if (!rq.params.name) {
rs.sendStatus(400);
return;
}
const room = await Rooms.getByName(rq.params.name.trim());
if (!room || rq.params.name == 'DormRoom') {
rs.sendStatus(404);
return;
} else {
rs.json(room.Room);
return;
}
},
);
route.router.post('/v1/roomRolePermissions',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
(rq, rs) => {
rs.sendStatus(200);
},
);

View File

@@ -0,0 +1,42 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import express from "express";
export const route = APIUtils.createRouter('/storefronts');
interface StorefrontFetchParams {
id?: string
}
route.router.get('/v3/giftdropstore/:id',
APIUtils.Authentication,
(rq: express.Request<StorefrontFetchParams>, rs: express.Response) => {
if (!rq.params.id) {
rs.sendStatus(400);
return;
}
rs.json({
StorefrontType: parseInt(rq.params.id),
NextUpdate: new Date(Date.now() + 604_800_000).toISOString(),
StoreItems: []
});
},
);

View File

@@ -23,8 +23,6 @@ route.router.get('/LoadingScreenTipData',
APIUtils.Authentication,
(_rq, rs) => {
rs.json([]);
}
APIUtils.emptyArrayResponse
);

View File

@@ -18,8 +18,10 @@ along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../apiutils.ts";
import { route as PlayerRoute } from "./match/player.ts";
import { route as GotoRoute } from "./match/goto.ts";
import { route as RoomRoute } from "./match/room.ts";
export const route = APIUtils.createRouter('/match');
route.router.use(PlayerRoute.path, PlayerRoute.router);
route.router.use(GotoRoute.path, GotoRoute.router);
route.router.use(GotoRoute.path, GotoRoute.router);
route.router.use(RoomRoute.path, RoomRoute.router);

View File

@@ -15,12 +15,15 @@ GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import Logging from "@proxnet/undead-logging";
import { APIUtils } from "../../apiutils.ts";
import Matchmaking from "../../data/live/base.ts";
import { MatchmakingErrorCode } from "../../data/live/types.ts";
import { AuthType } from "../../data/users.ts";
import express from "express";
const log = new Logging("MatchGotoRoute");
export const route = APIUtils.createRouter('/goto');
interface MatchmakingParams {
@@ -35,6 +38,7 @@ route.router.post('/room/:roomName',
async (rq: express.Request<MatchmakingParams>, rs: express.Response) => {
if (!rq.params.roomName) {
log.d('Matchmake failed: No room specified');
rs.json({
errorCode: MatchmakingErrorCode.NoSuchRoom
});

View File

@@ -23,6 +23,8 @@ import Presence, { PresenceExport } from "../../data/live/presence.ts";
import { AuthType } from "../../data/users.ts";
import Logging from "@proxnet/undead-logging";
import UnifiedProfile from "../../data/profiles.ts";
import { PlayerStatusVisibility, VRMovementMode } from "../../data/live/types.ts";
import { SettingKey } from "../../data/content/settings.ts";
const log = new Logging("MatchPlayerRoute");
@@ -51,7 +53,6 @@ route.router.get('/',
const presExport: PresenceExport[] = [];
for (const id of ids) {
const pres = await Presence.get(UnifiedProfile.get(id));
await pres.update();
presExport.push(await pres.export());
}
@@ -100,8 +101,51 @@ route.router.post('/heartbeat',
async (_rq, rs) => {
const pres = await Presence.get(rs.locals.profile);
await pres.update();
rs.json(await pres.export());
}
);
interface StatusVisibilityBody {
statusVisibility: PlayerStatusVisibility
}
const StatusVisibilitySchema = z.object({
statusVisibility: z.nativeEnum(PlayerStatusVisibility)
});
route.router.put('/statusvisibility',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
express.urlencoded({ extended: true }),
APIUtils.validateRequestBody(StatusVisibilitySchema),
async (rq: express.Request<NoBody, NoBody, StatusVisibilityBody>, rs: express.Response) => {
rs.locals.profile.Settings.setSetting(SettingKey.PlayerStatusVisibility, rq.body.statusVisibility.toString());
(await Presence.get(rs.locals.profile)).updateStatusVisibility();
rs.sendStatus(200);
},
);
interface VRMovementModeBody {
vrMovementMode: PlayerStatusVisibility
}
const VRMovementModeSchema = z.object({
vrMovementMode: z.nativeEnum(VRMovementMode)
});
route.router.put('/vrmovementmode',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
express.urlencoded({ extended: true }),
APIUtils.validateRequestBody(VRMovementModeSchema),
async (rq: express.Request<NoBody, NoBody, VRMovementModeBody>, rs: express.Response) => {
rs.locals.profile.Settings.setSetting(SettingKey.VRMovementMode, rq.body.vrMovementMode.toString());
(await Presence.get(rs.locals.profile)).updateVRMovementMode();
rs.sendStatus(200);
},
);

56
src/routes/match/room.ts Normal file
View File

@@ -0,0 +1,56 @@
/* Galvanic Corrosion - Rec Room custom server for communities.
<https://gitea.proxnet.dev/zombieb-galvanic-corrosion>
Copyright (C) 2025 @zombieb (Discord / proxnet Gitea)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>. */
import { APIUtils } from "../../apiutils.ts";
import Instances from "../../data/live/instances.ts";
import { AuthType } from "../../data/users.ts";
import express from "express";
export const route = APIUtils.createRouter('/room');
interface RoomGetInstancesParams {
id?: string
}
route.router.get('/:id/instances',
APIUtils.Authentication,
APIUtils.AuthenticationType(AuthType.Game),
(rq: express.Request<RoomGetInstancesParams>, rs: express.Response) => {
// TODO: send forbidden request when player is requesting instances for a room that they do not have permissions for
if (!rq.params.id) {
rs.sendStatus(400);
return;
}
const parsedId = parseInt(rq.params.id);
if (isNaN(parsedId)) {
rs.sendStatus(400);
return;
}
const instances = Instances.getAllRoomInstances(parsedId);
rs.json(instances.values().toArray().map(instance => ({
roomInstanceId: instance.roomInstanceId,
roomId: parsedId,
subRoomId: instance.subRoomId,
isFull: instance.isFull,
createdAt: new Date().toISOString(), // TODO: rewrite instance - create instance class rather than using sets - include datetime when instance was created
playerIds: Instances.getInstancePlayers(instance).values().toArray().map(profile => profile.getId())
})));
},
);

View File

@@ -21,6 +21,7 @@ import {
CompletionMessage,
Message,
MessageKind,
PushNotificationId,
SignalMessageType,
SignalRMessage,
SignalRMessageSchema,
@@ -32,6 +33,7 @@ import {
} from "./types.ts";
import { SocketTarget } from "./targets/targetbase.ts";
import { PlayerSocketSubscriptionTarget } from "./targets/SubscribeToPlayers.ts";
import Presence from "../data/live/presence.ts";
export class SignalRSocketHandler {
@@ -42,6 +44,8 @@ export class SignalRSocketHandler {
#Targets: Map<string, SocketTarget> = new Map();
#PresenceUpdateId: number;
constructor(socket: WebSocket, player: Profile) {
this.#socket = socket;
@@ -53,6 +57,11 @@ export class SignalRSocketHandler {
this.#Targets.set('SubscribeToPlayers', new PlayerSocketSubscriptionTarget());
this.#PresenceUpdateId = setInterval(async () => {
const pres = await Presence.get(this.#profile);
this.sendNotification("PresenceUpdate", await pres.export());
}, 8000);
}
async #dispatchTarget<T = unknown>(target: string, args: unknown): Promise<TargetResult> {
@@ -136,6 +145,7 @@ export class SignalRSocketHandler {
destroy(sock: SignalRSocketHandler) {
return () => {
clearInterval(sock.#PresenceUpdateId);
sock.sendRaw({ type: 7, error: "Socket closed" });
sock.#socket.close();
sock.#log.i(`Closed hub socket`);
@@ -149,4 +159,16 @@ export class SignalRSocketHandler {
this.#log.d(`SERVER MESSAGE\n ${type}\n ${JSON.stringify(data as SignalRMessage)}`);
}
sendNotification(id: PushNotificationId | string, args: object) {
const msg: SignalRMessage = {
type: SignalMessageType.Invocation,
target: "Notification",
arguments: [JSON.stringify({
Id: id,
Msg: args
})]
}
this.sendRaw(msg);
}
}

View File

@@ -144,4 +144,42 @@ export interface TargetResultFailure extends TargetResultBase {
export interface TargetResultNotATarget extends TargetResultBase {
type: TargetResultType.NotATarget
}
export type TargetResult = TargetResultSuccess | TargetResultFailure | TargetResultNotATarget;
export type TargetResult = TargetResultSuccess | TargetResultFailure | TargetResultNotATarget;
export enum PushNotificationId {
RelationshipChanged = 1,
MessageReceived,
MessageDeleted,
PresenceHeartbeatResponse,
RefreshLogin,
Logout,
SubscriptionUpdateProfile = 11,
SubscriptionUpdatePresence,
SubscriptionUpdateGameSession,
SubscriptionUpdateRoom = 15,
ModerationQuitGame = 20,
ModerationUpdateRequired,
ModerationKick,
ModerationKickAttemptFailed,
ModerationRoomBan,
ServerMaintenance,
GiftPackageReceived = 30,
GiftPackageReceivedImmediate,
ProfileJuniorStatusUpdate = 40,
RelationshipsInvalid = 50,
StorefrontBalanceAdd = 60,
StorefrontBalanceUpdate,
StorefrontBalancePurchase,
ConsumableMappingAdded = 70,
ConsumableMappingRemoved,
PlayerEventCreated = 80,
PlayerEventUpdated,
PlayerEventDeleted,
PlayerEventResponseChanged,
PlayerEventResponseDeleted,
PlayerEventStateChanged,
ChatMessageReceived = 90,
CommunityBoardUpdate = 95,
CommunityBoardAnnouncementUpdate,
InventionModerationStateChanged = 100
}