Skip to content

Latest commit

 

History

History
886 lines (767 loc) · 22.1 KB

json-examples.md

File metadata and controls

886 lines (767 loc) · 22.1 KB

Structure

Every Dink POST request is a multipart body.

In particular, the Content-Type is multipart/form-data to accomodate the below JSON and optional screenshots in accordance with the Discord API specification.

Thus, any third-party consumer should utilize the body entity named payload_json to access the relevant JSON object. The optional body entity for the screenshot is named file, and the underlying data stream (which should not exceed 8MB) can be image/png or image/jpeg (less common).

Due to this structure, trying to parse the full multipart/form-data as JSON will not succeed until you specifically grab the payload_json entity. Competent web frameworks should handle the multipart parsing, so you can easily access the relevant form values.

See here for an example project that leverages @fastify/multipart to read the JSON payload and screenshot file.
For Cloudflare Workers, utilize the request.formData() method.
For Express, utilize the Multer middleware.
For Golang, utilize the ParseMultipartForm function.
For http4k, utilize the http4k-multipart module.
For Ktor, utilize the receiveMultipart method.
For Jooby, utilize the form method.
For Vert.x-Web, utilize the formAttributes method.
For Quarkus, utilize the @RestForm annotation.
For Spring, utilize the @RequestPart annotation.

In the examples below, content is populated instead of embeds for simplicity; this would correspond to the advanced setting 'Use Rich Embeds' being disabled. Third-party integrations should rely on the extra object instead of content/embeds.

All

JSON sent with every notification:

{
  "content": "Text message as set by the user",
  "extra": {},
  "type": "NOTIFICATION_TYPE",
  "playerName": "your rsn",
  "accountType": "NORMAL | IRONMAN | HARDCORE_IRONMAN",
  "seasonalWorld": "true | false",
  "dinkAccountHash": "abcdefghijklmnopqrstuvwxyz1234abcdefghijklmnopqrstuvwxyz",
  "embeds": []
}

JSON sent with every notification but only in certain circumstances:

{
  "clanName": "Dink QA",
  "groupIronClanName": "Dink QA",
  "discordUser": {
    "id": "012345678910111213",
    "name": "Gamer",
    "avatarHash": "abc123def345abc123def345abc123de"
  },
  "world": 518,
  "regionId": 12850
}

clanName is only sent when the player is in a clan and has the advanced setting Send Clan Name enabled.
groupIronClanName is only sent when the player is a GIM and has the advanced setting Send GIM Clan Name enabled.
The discordUser object is only sent when Discord is open and the advanced setting Send Discord Profile is enabled.
world and regionId are only sent when the advanced setting Include Location is enabled (default: true).

Note: The examples below omit playerName, accountType, and dinkAccountHash keys because they are always the same.

Deaths

JSON for non-combat death:

{
  "content": "%USERNAME% has died...",
  "extra": {
    "valueLost": 300,
    "isPvp": false,
    "keptItems": [],
    "lostItems": [
      {
        "id": 314,
        "quantity": 100,
        "priceEach": 3,
        "name": "Feather"
      }
    ],
    "location": {
      "regionId": 10546,
      "plane": 0,
      "instanced": false
    }
  },
  "type": "DEATH"
}

JSON for PvP scenarios:

{
  "content": "%USERNAME% has just been PKed by %PKER% for %VALUELOST% gp...",
  "extra": {
    "valueLost": 300,
    "isPvp": true,
    "killerName": "%PKER%",
    "keptItems": [],
    "lostItems": [
      {
        "id": 314,
        "quantity": 100,
        "priceEach": 3,
        "name": "Feather"
      }
    ],
    "location": {
      "regionId": 10546,
      "plane": 0,
      "instanced": false
    }
  },
  "type": "DEATH"
}

JSON for NPC scenarios:

{
  "content": "%USERNAME% has died...",
  "extra": {
    "valueLost": 300,
    "isPvp": false,
    "killerName": "%NPC%",
    "killerNpcId": 69,
    "keptItems": [],
    "lostItems": [
      {
        "id": 314,
        "quantity": 100,
        "priceEach": 3,
        "name": "Feather"
      }
    ],
    "location": {
      "regionId": 10546,
      "plane": 0,
      "instanced": false
    }
  },
  "type": "DEATH"
}

Collection

JSON for Collection Notifications:

{
  "content": "%USERNAME% has added %ITEM% to their collection",
  "extra": {
    "itemName": "Zamorak chaps",
    "itemId": 10372,
    "price": 500812,
    "completedEntries": 420,
    "totalEntries": 1443,
    "dropperName": "Clue Scroll (Hard)",
    "dropperType": "EVENT",
    "dropperKillCount": 1500
  },
  "type": "COLLECTION"
}

Note: dropperName/dropperType/dropperKillCount may not be present for all collection log notifications.

Level

JSON for Levelups:

{
  "content": "%USERNAME% has levelled %SKILL%",
  "extra": {
    "levelledSkills": {
      // These are the skills that dinked
      "Skill name": 30
    },
    "allSkills": {
      // These are all the skills
      "Skill name": 30,
      "Other skill": 1
    },
    "combatLevel": {
      "value": 50,
      "increased": false
    }
  },
  "type": "LEVEL"
}

Note: Level 127 in JSON corresponds to attaining max experience in a skill (200M).

JSON for XP Milestones:

{
  "content": "%USERNAME% has levelled %SKILL%",
  "extra": {
    "xpData": {
      // The XP for each skill
      "Skill one": 1234567,
      "Skill two": 2345678,
      "Skill three": 3456789
    },
    "milestoneAchieved": ["Skill name"], // The skill(s) that hit a milestone
    "interval": 5000000 // The configured XP interval
  },
  "type": "XP_MILESTONE"
}

Loot

JSON for Loot Notifications:

{
  "content": "%USERNAME% has looted: \n\n%LOOT%\nFrom: %SOURCE%",
  "extra": {
    "items": [
      {
        "id": 1234,
        "quantity": 1,
        "priceEach": 42069,
        "name": "Some item",
        "criteria": ["VALUE"],
        "rarity": null
      }
    ],
    "source": "Tombs of Amascut",
    "party": ["%USERNAME%", "another RSN", "yet another RSN"],
    "category": "EVENT",
    "killCount": 60,
    "rarestProbability": 0.001,
    "npcId": null
  },
  "type": "LOOT"
}

The possible values for extra.category correspond to the LootRecordType enum.

The possible values for extra.items[].criteria correspond to our LootCriteria enum.

killCount is only specified for NPC/EVENT loot with the base RuneLite Loot Tracker plugin enabled.

rarity is currently only populated for NPC drops (and some pickpocket events). This data is (imperfectly) scraped from the wiki, so it may not be 100% accurate. Also, we do not report a rarity if the NPC always drops the item on every kill.

The items are valued at GE prices (when possible) if the user has not disabled the Use actively traded price base RuneLite setting. Otherwise, the store price of the item is used.

The extra.party field is only populated for raids loot (i.e., COX, TOA, TOB).

The extra.npcId field is populated only if the category is NPC or PICKPOCKET.

Slayer

JSON for Slayer Notifications:

{
  "content": "%USERNAME% has completed a slayer task: %TASK%, getting %POINTS% points and making that %TASKCOUNT% tasks completed",
  "extra": {
    "slayerTask": "Slayer task name",
    "slayerCompleted": "30",
    "slayerPoints": "15",
    "killCount": 135,
    "monster": "Kalphite"
  },
  "type": "SLAYER"
}

Quests

JSON for Quest Notifications:

{
  "content": "%USERNAME% has completed a quest: %QUEST%",
  "extra": {
    "questName": "Dragon Slayer I",
    "completedQuests": 22,
    "totalQuests": 156,
    "questPoints": 44,
    "totalQuestPoints": 293
  },
  "type": "QUEST"
}

Clue

JSON for Clue Notifications:

{
  "content": "%USERNAME% has completed a %CLUE% clue, they have completed %COUNT%.\nThey obtained:\n\n%LOOT%",
  "extra": {
    "clueType": "Beginner",
    "numberCompleted": 123,
    "items": [
      {
        "id": 1234,
        "quantity": 1,
        "priceEach": 42069,
        "name": "Some item"
      }
    ]
  },
  "type": "CLUE"
}

The items are valued at GE prices (when possible) if the user has not disabled the Use actively traded price base RuneLite setting. Otherwise, the store price of the item is used.

Kill Count

JSON for Kill Count Notifications:

{
  "content": "%USERNAME% has defeated %BOSS% with a completion count of %COUNT%",
  "extra": {
    "boss": "Chambers of Xeric",
    "count": 69,
    "gameMessage": "Your completed Chambers of Xeric count is: 69.",
    "time": "PT46M34S",
    "isPersonalBest": true,
    "party": ["%USERNAME%", "another RSN", "yet another RSN"]
  },
  "type": "KILL_COUNT"
}

When an associated duration is not found, extra.time and extra.isPersonalBest are not populated.

Note: when boss is Penance Queen, count refers to the high level gamble count, rather than kill count.

Also, extra.party is only populated for certain bosses (see Loot for more details).

Combat Achievements

JSON for Combat Achievement Notifications:

{
  "content": "%USERNAME% has completed %TIER% combat task: %TASK%",
  "extra": {
    "tier": "GRANDMASTER",
    "task": "Peach Conjurer",
    "taskPoints": 6,
    "totalPoints": 1337,
    "tierProgress": 517,
    "tierTotalPoints": 645,
    "totalPossiblePoints": 14814,
    "currentTier": "MASTER",
    "nextTier": "GRANDMASTER"
  },
  "type": "COMBAT_ACHIEVEMENT"
}

JSON for Combat Achievement Tier Completion Notifications:

{
  "content": "%USERNAME% has unlocked the rewards for the %COMPLETED% tier, by completing the combat task: %TASK%",
  "extra": {
    "tier": "GRANDMASTER",
    "task": "Peach Conjurer",
    "taskPoints": 6,
    "totalPoints": 1465,
    "tierProgress": 0,
    "tierTotalPoints": 540,
    "totalPossiblePoints": 14814,
    "nextTier": "GRANDMASTER",
    "justCompletedTier": "MASTER"
  },
  "type": "COMBAT_ACHIEVEMENT"
}

Achievement Diary

JSON for Achievement Diary Notifications:

{
  "content": "%USERNAME% has completed the %DIFFICULTY% %AREA% Achievement Diary, for a total of %TOTAL% diaries completed",
  "extra": {
    "area": "Varrock",
    "difficulty": "HARD",
    "total": 15,
    "tasksCompleted": 152,
    "tasksTotal": 492,
    "areaTasksCompleted": 37,
    "areaTasksTotal": 42
  },
  "type": "ACHIEVEMENT_DIARY"
}

Pets

JSON for Pet Notifications:

{
  "content": "%USERNAME% has a funny feeling they are being followed",
  "extra": {
    "petName": "Ikkle hydra",
    "milestone": "5,000 killcount",
    "duplicate": false
  },
  "type": "PET"
}

petName is only included if the game sent it to the users chat via untradeable drop, collection log, or clan notifications.
milestone is only included if a clan notification was triggered.

Speedrunning

JSON for Personal Best Speedrun Notifications:

{
  "content": "%USERNAME% has just beat their personal best in a speedrun of %QUEST% with a time of %TIME%",
  "extra": {
    "questName": "Cook's Assistant",
    "personalBest": "1:13.20",
    "currentTime": "1:13.20",
    "isPersonalBest": true
  },
  "type": "SPEEDRUN"
}

JSON for Normal Speedrun Notifications:

{
  "content": "%USERNAME% has just finished a speedrun of %QUEST% with a time of %TIME% (their PB is %BEST%)",
  "extra": {
    "questName": "Cook's Assistant",
    "personalBest": "1:13.20",
    "currentTime": "1:22.20"
  },
  "type": "SPEEDRUN"
}

BA Gambles

JSON for BA Gambles Notifications:

{
  "content": "%USERNAME% has reached %COUNT% high gambles",
  "extra": {
    "gambleCount": 500,
    "items": [
      {
        "id": 3122,
        "quantity": 1,
        "priceEach": 35500,
        "name": "Granite shield"
      }
    ]
  },
  "type": "BARBARIAN_ASSAULT_GAMBLE"
}

Player Kills

JSON for PK Notifications:

{
  "content": "%USERNAME% has PK'd %TARGET%",
  "type": "PLAYER_KILL",
  "extra": {
    "victimName": "%TARGET%",
    "victimCombatLevel": 69,
    "victimEquipment": {
      "AMULET": {
        "id": 1731,
        "priceEach": 1987,
        "name": "Amulet of power"
      },
      "WEAPON": {
        "id": 1333,
        "priceEach": 14971,
        "name": "Rune scimitar"
      },
      "TORSO": {
        "id": 1135,
        "priceEach": 4343,
        "name": "Green d'hide body"
      },
      "LEGS": {
        "id": 1099,
        "priceEach": 2077,
        "name": "Green d'hide chaps"
      },
      "HANDS": {
        "id": 1065,
        "priceEach": 1392,
        "name": "Green d'hide vambraces"
      }
    },
    "world": 394,
    "location": {
      "x": 3334,
      "y": 4761,
      "plane": 0
    },
    "myHitpoints": 20,
    "myLastDamage": 12
  }
}

world and location are not sent if the user has disabled the "Include Location" notifier setting.

Group Storage

JSON for GIM Bank Notifications:

{
  "content": "%USERNAME% has deposited: %DEPOSITED% | %USERNAME% has withdrawn: %WITHDRAWN%",
  "type": "GROUP_STORAGE",
  "accountType": "GROUP_IRONMAN | HARDCORE_GROUP_IRONMAN",
  "extra": {
    "groupName": "group name",
    "deposits": [
      {
        "id": 315,
        "name": "Shrimps",
        "quantity": 2,
        "priceEach": 56
      },
      {
        "id": 1205,
        "name": "Bronze dagger",
        "quantity": 1,
        "priceEach": 53
      }
    ],
    "withdrawals": [
      {
        "id": 1265,
        "name": "Bronze pickaxe",
        "quantity": 1,
        "priceEach": 22
      }
    ],
    "netValue": 143
  }
}

accountType is always GROUP_IRONMAN or HARDCORE_GROUP_IRONMAN for Group Storage notifications.

Grand Exchange

JSON for GE Notifications:

{
  "content": "%USERNAME% %TYPE% %ITEM% on the GE",
  "type": "GRAND_EXCHANGE",
  "extra": {
    "slot": 1,
    "status": "SOLD",
    "item": {
      "id": 314,
      "quantity": 2,
      "priceEach": 3,
      "name": "Feather"
    },
    "marketPrice": 2,
    "targetPrice": 3,
    "targetQuantity": 2,
    "sellerTax": 0
  }
}

Unlike GrandExchangeOfferChanged#getSlot, extra.slot is one-indexed; values can range from 1 to 8 (inclusive) for members, and 1 to 3 (inclusive) for F2P.

See javadocs for the possible values of extra.status.

Player Trades

JSON for Player Trade Notifications:

{
  "content": "%USERNAME% traded with %COUNTERPARTY%",
  "type": "TRADE",
  "extra": {
    "counterparty": "%COUNTERPARTY%",
    "receivedItems": [
      {
        "id": 314,
        "quantity": 100,
        "priceEach": 2,
        "name": "Feather"
      }
    ],
    "givenItems": [
      {
        "id": 2,
        "quantity": 3,
        "priceEach": 150,
        "name": "Cannonball"
      }
    ],
    "receivedValue": 200,
    "givenValue": 450
  }
}

Leagues

JSON for Area Unlock Notifications:

{
  "type": "LEAGUES_AREA",
  "content": "%USERNAME% selected their second region: Kandarin.",
  "accountType": "IRONMAN",
  "seasonalWorld": true,
  "extra": {
    "area": "Kandarin",
    "index": 2,
    "tasksCompleted": 200,
    "tasksUntilNextArea": 200
  }
}

Note: index refers to the order of region unlocks.
Here, Kandarin was the second region selected.
For all players, Karamja is the zeroth region selected (and there is no notification for Misthalin).

JSON for Combat Mastery Unlocked Notification:

{
  "type": "LEAGUES_MASTERY",
  "content": "%USERNAME% unlocked a new Combat Mastery: %MASTERY%.",
  "accountType": "IRONMAN",
  "seasonalWorld": true,
  "extra": {
    "masteryType": "Melee",
    "masteryTier": 1
  }
}

JSON for Relic Chosen Notifications:

{
  "type": "LEAGUES_RELIC",
  "content": "%USERNAME% unlocked a Tier 1 Relic: Production Prodigy.",
  "accountType": "IRONMAN",
  "seasonalWorld": true,
  "extra": {
    "relic": "Production Prodigy",
    "tier": 1,
    "requiredPoints": 0,
    "totalPoints": 20,
    "pointsUntilNextTier": 480
  }
}

JSON for Task Completed Notifications:

{
  "type": "LEAGUES_TASK",
  "content": "%USERNAME% completed a Easy task: Pickpocket a Citizen.",
  "accountType": "IRONMAN",
  "seasonalWorld": true,
  "extra": {
    "taskName": "Pickpocket a Citizen",
    "difficulty": "EASY",
    "taskPoints": 10,
    "totalPoints": 30,
    "tasksCompleted": 3,
    "tasksUntilNextArea": 57,
    "pointsUntilNextRelic": 470,
    "pointsUntilNextTrophy": 2470
  }
}

JSON for Task Notifications that unlocked a Trophy:

{
  "type": "LEAGUES_TASK",
  "content": "%USERNAME% completed a Hard task, The Frozen Door, unlocking the Bronze trophy!",
  "accountType": "IRONMAN",
  "seasonalWorld": true,
  "extra": {
    "taskName": "The Frozen Door",
    "difficulty": "HARD",
    "taskPoints": 80,
    "totalPoints": 2520,
    "tasksCompleted": 119,
    "tasksUntilNextArea": 81,
    "pointsUntilNextRelic": 1480,
    "pointsUntilNextTrophy": 2480,
    "earnedTrophy": "Bronze"
  }
}

Note: Fields like tasksUntilNextArea, pointsUntilNextRelic, and pointsUntilNextTrophy can be omitted if there is no next level of progression (i.e., all three regions selected, all relic tiers unlocked, all trophies acquired).

accountType is always IRONMAN for Leagues, unrelated to what the user is outside of Leagues.

Chat

JSON for Matching Chat Message Notifications:

{
  "type": "CHAT",
  "content": "%USERNAME% received a chat message: `You've been playing for a while, consider taking a break from your screen.`",
  "extra": {
    "type": "GAMEMESSAGE",
    "message": "You've been playing for a while, consider taking a break from your screen.",
    "source": null,
    "clanTitle": null
  }
}

Note: The possible values for extra.type are documented in RuneLite's javadocs.

When extra.type corresponds to a player-sent message (e.g., PUBLICCHAT, PRIVATECHAT, FRIENDSCHAT, CLAN_CHAT, CLAN_GUEST_CHAT), the extra.source value is set to the player's name that sent the message.

When extra.type is UNKNOWN, the extra.source value is set to the originating runelite event (e.g., CommandExecuted, NotificationFired).

When extra.type is CLAN_CHAT or CLAN_GUEST_CHAT or CLAN_GIM_CHAT or CLAN_MESSAGE (only for user joins), the extra.clanTitle object includes the clan rank id (integer) and title name (string), corresponding to RuneLite's ClanTitle class.

Metadata

JSON for Login Notifications:

{
  "content": "%USERNAME% logged into World %WORLD%",
  "type": "LOGIN",
  "extra": {
    "world": 338,
    "collectionLog": {
      "completed": 651,
      "total": 1477
    },
    "combatAchievementPoints": {
      "completed": 503,
      "total": 2005
    },
    "achievementDiary": {
      "completed": 42,
      "total": 48
    },
    "achievementDiaryTasks": {
      "completed": 477,
      "total": 492
    },
    "barbarianAssault": {
      "highGambleCount": 0
    },
    "skills": {
      "totalExperience": 346380298,
      "totalLevel": 2164,
      "levels": {
        "Hunter": 90,
        "Thieving": 86,
        "Runecraft": 86,
        "Construction": 86,
        "Cooking": 103,
        "Magic": 106,
        "Fletching": 99,
        "Herblore": 91,
        "Firemaking": 100,
        "Attack": 107,
        "Fishing": 92,
        "Crafting": 96,
        "Hitpoints": 111,
        "Ranged": 110,
        "Mining": 88,
        "Smithing": 91,
        "Agility": 82,
        "Woodcutting": 96,
        "Slayer": 104,
        "Defence": 103,
        "Strength": 104,
        "Prayer": 91,
        "Farming": 100
      },
      "experience": {
        "Hunter": 5420696,
        "Thieving": 3696420,
        "Runecraft": 3969420,
        "Construction": 3680085,
        "Cooking": 19696420,
        "Magic": 28008135,
        "Fletching": 13696420,
        "Herblore": 5969420,
        "Firemaking": 14420666,
        "Attack": 30696420,
        "Fishing": 6632248,
        "Crafting": 9696420,
        "Hitpoints": 46969666,
        "Ranged": 42069420,
        "Mining": 4696420,
        "Smithing": 6428696,
        "Agility": 2666420,
        "Woodcutting": 9696666,
        "Slayer": 21420696,
        "Defence": 21212121,
        "Strength": 23601337,
        "Prayer": 6369666,
        "Farming": 15666420
      }
    },
    "questCount": {
      "completed": 156,
      "total": 158
    },
    "questPoints": {
      "completed": 296,
      "total": 300
    },
    "slayer": {
      "points": 2204,
      "streak": 1074
    },
    "pets": [
      {
        "itemId": 11995,
        "name": "Pet chaos elemental"
      },
      {
        "itemId": 13071,
        "name": "Chompy chick"
      }
    ]
  }
}

extra.pets requires the base Chat Commands plugin to be enabled.
collectionLog data can be missing if the user does not have the Character Summary tab selected (since the client otherwise is not sent that data).

JSON for Logout Notifications:

{
  "type": "LOGOUT",
  "content": "%USERNAME% logged out"
}