parse.go 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. package gen7
  2. import (
  3. "bytes"
  4. "encoding/binary"
  5. "fmt"
  6. "strings"
  7. "time"
  8. "unicode/utf16"
  9. p "github.com/ajswis/go-pkparse-server/pokemon"
  10. )
  11. type consoleRegion map[uint8]string
  12. // Parse accepts a raw pokemon (byte slice) and either returns a Pokemon struct
  13. // with all applicable fields populated or an error depicting the reason for
  14. // failure.
  15. func Parse(rawPokemon p.RawPokemon) (*p.Pokemon, error) {
  16. if err := validateRawPokemon(rawPokemon); err != nil {
  17. return nil, err
  18. }
  19. consoleRegions := consoleRegion{0x00: "J", 0x01: "U", 0x02: "E", 0x03: "?", 0x04: "C", 0x05: "K", 0x06: "T"}
  20. var pkmn p.Pokemon
  21. pkmn.RawPokemon = &rawPokemon
  22. // Block A
  23. pkmn.EncryptionConstant = binary.LittleEndian.Uint32(rawPokemon[0x00:0x08])
  24. pkmn.PokedexNumber = toUint16(rawPokemon[0x08:0x0a])
  25. pkmn.HeldItemID = toUint16(rawPokemon[0x0a:0x0c])
  26. pkmn.TrainerID = toUint16(rawPokemon[0x0c:0x0e])
  27. pkmn.SecretID = toUint16(rawPokemon[0x0e:0x10])
  28. pkmn.FullTrainerID = binary.LittleEndian.Uint32(rawPokemon[0x0c:0x10])
  29. pkmn.Experience = binary.LittleEndian.Uint32(rawPokemon[0x10:0x15])
  30. pkmn.AbilityID = rawPokemon[0x14]
  31. pkmn.AbilityNum = rawPokemon[0x14]
  32. pkmn.PID = binary.LittleEndian.Uint32(rawPokemon[0x18:0x1c])
  33. pkmn.NatureID = rawPokemon[0x1c]
  34. pkmnGenderByte := rawPokemon[0x1d]
  35. pkmn.FatefulEncounter = (pkmnGenderByte & 0x01) == 1
  36. if (pkmnGenderByte & 0x02) == 0x02 {
  37. pkmn.Gender = "F"
  38. } else if (pkmnGenderByte & 0x04) == 0x04 {
  39. pkmn.Gender = ""
  40. } else {
  41. pkmn.Gender = "M"
  42. }
  43. pkmn.FormID = pkmnGenderByte >> 3
  44. pkmn.HPEV = rawPokemon[0x1e]
  45. pkmn.AtkEV = rawPokemon[0x1f]
  46. pkmn.DefEV = rawPokemon[0x20]
  47. pkmn.SpeEV = rawPokemon[0x21]
  48. pkmn.SpAtkEV = rawPokemon[0x22]
  49. pkmn.SpDefEV = rawPokemon[0x23]
  50. pokerusByte := rawPokemon[0x2b]
  51. pkmn.PokerusDuration = pokerusByte & 15
  52. pkmn.PokerusStrain = pokerusByte >> 4
  53. // Block B
  54. pkmn.RawNickname = rawPokemon[0x40:0x58]
  55. pkmn.Nickname = binaryToUTF16leString(pkmn.RawNickname)
  56. pkmn.Move1ID = toUint16(rawPokemon[0x5a:0x5c])
  57. pkmn.Move2ID = toUint16(rawPokemon[0x5c:0x5e])
  58. pkmn.Move3ID = toUint16(rawPokemon[0x5e:0x60])
  59. pkmn.Move4ID = toUint16(rawPokemon[0x60:0x62])
  60. pkmn.Move1PP = rawPokemon[0x62]
  61. pkmn.Move2PP = rawPokemon[0x63]
  62. pkmn.Move3PP = rawPokemon[0x64]
  63. pkmn.Move4PP = rawPokemon[0x65]
  64. pkmn.Move1PPUsed = rawPokemon[0x66]
  65. pkmn.Move2PPUsed = rawPokemon[0x67]
  66. pkmn.Move3PPUsed = rawPokemon[0x68]
  67. pkmn.Move4PPUsed = rawPokemon[0x69]
  68. pkmn.EggMove1ID = toUint16(rawPokemon[0x6a:0x6c])
  69. pkmn.EggMove2ID = toUint16(rawPokemon[0x6c:0x6e])
  70. pkmn.EggMove3ID = toUint16(rawPokemon[0x6e:0x70])
  71. pkmn.EggMove4ID = toUint16(rawPokemon[0x70:0x72])
  72. ivBytes := binary.LittleEndian.Uint32(rawPokemon[0x74:0x78])
  73. pkmn.HPIV = ivBytes & 0x1f
  74. pkmn.AtkIV = ivBytes >> 5 & 0x1f
  75. pkmn.DefIV = ivBytes >> 10 & 0x1f
  76. pkmn.SpeIV = ivBytes >> 15 & 0x1f
  77. pkmn.SpAtkIV = ivBytes >> 20 & 0x1f
  78. pkmn.SpDefIV = ivBytes >> 25 & 0x1f
  79. pkmn.IsEgg = ((ivBytes >> 30) % 2) == 1
  80. pkmn.IsNicknamed = ((ivBytes >> 31) % 2) == 1
  81. // Block C
  82. pkmn.RawNotOTName = rawPokemon[0x78:0x90]
  83. pkmn.NotOTName = binaryToUTF16leString(pkmn.RawNotOTName)
  84. if rawPokemon[0x92] == 1 {
  85. pkmn.NotOTGender = "F"
  86. } else {
  87. pkmn.NotOTGender = "M"
  88. }
  89. pkmn.CurrentHandlerIsOT = rawPokemon[0x93] == 1
  90. pkmn.GeoLocation1RegionID = rawPokemon[0x94]
  91. pkmn.GeoLocation1CountryID = rawPokemon[0x95]
  92. pkmn.GeoLocation2RegionID = rawPokemon[0x96]
  93. pkmn.GeoLocation2CountryID = rawPokemon[0x97]
  94. pkmn.GeoLocation3RegionID = rawPokemon[0x98]
  95. pkmn.GeoLocation3CountryID = rawPokemon[0x99]
  96. pkmn.GeoLocation4RegionID = rawPokemon[0x9a]
  97. pkmn.GeoLocation4CountryID = rawPokemon[0x9b]
  98. pkmn.GeoLocation5RegionID = rawPokemon[0x9c]
  99. pkmn.GeoLocation5CountryID = rawPokemon[0x9d]
  100. pkmn.NotOTFriendship = rawPokemon[0xa2]
  101. pkmn.NotOTAffection = rawPokemon[0xa3]
  102. pkmn.NotOTMemoryIntensity = rawPokemon[0xa4]
  103. pkmn.NotOTMemoryLine = rawPokemon[0xa5]
  104. pkmn.NotOTMemoryFeeling = rawPokemon[0xa6]
  105. pkmn.NotOTMemoryTextVar = toUint16(rawPokemon[0xa8:0xaa])
  106. pkmn.Fullness = rawPokemon[0xae]
  107. pkmn.Enjoyment = rawPokemon[0xaf]
  108. // Block D
  109. pkmn.RawOTName = rawPokemon[0xb0:0xc8]
  110. pkmn.OTName = binaryToUTF16leString(pkmn.RawOTName)
  111. pkmn.OTFriendship = rawPokemon[0xca]
  112. pkmn.OTAffection = rawPokemon[0xcb]
  113. pkmn.OTMemoryIntensity = rawPokemon[0xcc]
  114. pkmn.OTMemoryLine = rawPokemon[0xcd]
  115. pkmn.OTMemoryTextVar = toUint16(rawPokemon[0xce:0xd0])
  116. pkmn.OTMemoryFeeling = rawPokemon[0xd0]
  117. pkmn.DateEggReceived = uint16ToDatetime(binary.LittleEndian.Uint32(rawPokemon[0xd0:0xd4]))
  118. pkmn.DateMet = uint16ToDatetime(binary.LittleEndian.Uint32(rawPokemon[0xd4:0xd8]))
  119. pkmn.EggLocationID = toUint16(rawPokemon[0xd8:0xda])
  120. pkmn.MetLocationID = toUint16(rawPokemon[0xda:0xdc])
  121. pkmn.PokeballID = rawPokemon[0xdc]
  122. pkmn.LevelMet = rawPokemon[0xdd] & 0x7f
  123. if otGenderBit := rawPokemon[0xdd] >> 7; otGenderBit == 1 {
  124. pkmn.OTGender = "F"
  125. } else {
  126. pkmn.OTGender = "M"
  127. }
  128. pkmn.Gen4EncounterTypeID = rawPokemon[0xde]
  129. pkmn.OTGameID = rawPokemon[0xdf]
  130. pkmn.CountryID = rawPokemon[0xe0]
  131. pkmn.RegionID = rawPokemon[0xe1]
  132. pkmn.RawConsoleRegion = rawPokemon[0xe2]
  133. pkmn.ConsoleRegion = consoleRegions[pkmn.RawConsoleRegion]
  134. pkmn.OTLanguageID = rawPokemon[0xe3]
  135. return &pkmn, nil
  136. }
  137. func binaryToUTF16leString(b []byte) string {
  138. r := bytes.NewReader(b)
  139. u16 := make([]uint16, len(b)/2)
  140. binary.Read(r, binary.LittleEndian, &u16)
  141. runes := utf16.Decode(u16)
  142. s := strings.Split(string(runes), "\u0000")[0]
  143. return s
  144. }
  145. func calculateChecksum(b []byte) uint16 {
  146. var sum uint16
  147. for i := 8; i < 232; i += 2 {
  148. sum += toUint16(b[i : i+2])
  149. }
  150. return sum & 0xffff
  151. }
  152. func toUint16(b []byte) uint16 {
  153. return binary.LittleEndian.Uint16(b)
  154. }
  155. func validateRawPokemon(rawPokemon p.RawPokemon) error {
  156. if l := len(rawPokemon); l != 232 && l != 260 {
  157. return fmt.Errorf("Invalid length for generation 7 pokemon: %d bytes", l)
  158. }
  159. checksumValue := toUint16(rawPokemon[0x06:0x08])
  160. if calculatedChecksum := calculateChecksum(rawPokemon); calculatedChecksum != checksumValue {
  161. return fmt.Errorf("Invalid checksum for generation 7 pokemon")
  162. }
  163. if toUint16(rawPokemon[0x04:0x06]) == 1 || rawPokemon[0x58] == 1 || rawPokemon[0x90] == 1 || rawPokemon[0xc8] == 1 {
  164. return fmt.Errorf("Invalid bytes supplied for generation 7 pokemon")
  165. }
  166. return nil
  167. }
  168. func uint16ToDatetime(i uint32) int64 {
  169. if i == 0 {
  170. return 0
  171. }
  172. year := int((i & 0xff) + 2000)
  173. month := time.Month(int((i >> 8) & 0xff))
  174. day := int((i >> 16) & 0xff)
  175. date := time.Date(year, month, day, 0, 0, 0, 0, time.UTC)
  176. return date.Unix()
  177. }