notes.js 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. const fileHash = require('../../util/backup_filehash')
  2. const log = require('../../util/log')
  3. const NOTES_DB = fileHash('Library/Notes/notes.sqlite')
  4. const NOTES2_DB = fileHash('NoteStore.sqlite', 'AppDomainGroup-group.com.apple.notes')
  5. module.exports = {
  6. version: 4,
  7. name: 'notes',
  8. description: `List all iOS notes`,
  9. requiresBackup: true,
  10. // Run on a v3 lib / backup object.
  11. run (lib, { backup }) {
  12. return getAllNotes(backup)
  13. },
  14. // Public facing properties
  15. output: {
  16. id: el => el.Z_PK,
  17. identifier: el => el.ZIDENTIFIER,
  18. modified: el => (el.XFORMATTEDDATESTRING || el.XFORMATTEDDATESTRING1) + '',
  19. passwordProtected: el => !!el.ZISPASSWORDPROTECTED,
  20. title: el => (el.ZTITLE || el.ZTITLE1 || el.ZTITLE2 || '').trim() || null,
  21. content: el => el.ZCONTENT || null
  22. }
  23. }
  24. function getAllNotes (backup) {
  25. return new Promise(async (resolve, reject) => {
  26. var newNotes
  27. // Try iOS 10/11 query.
  28. try {
  29. newNotes = await getNewNotesiOS10iOS11(backup)
  30. } catch (e) {
  31. log.verbose(`couldn't query notes as iOS10/11, trying iOS9`, e)
  32. }
  33. // If iOS 10/11 query fails, try iOS 9.
  34. if (newNotes == null) {
  35. try {
  36. newNotes = await getNewNotesiOS9(backup)
  37. } catch (e) {
  38. log.verbose(`couldn't query notes as iOS9`, e)
  39. }
  40. }
  41. // Try to fetch old notes database
  42. try {
  43. var oldNotes = await getOldNotes(backup)
  44. } catch (e) {
  45. log.verbose(`couldn't query old notes`, e)
  46. }
  47. // If we didn't get anything successfully, reject.
  48. if (newNotes == null && oldNotes == null) {
  49. return reject(new Error(`Couldn't find any known notes database in the system.`))
  50. }
  51. // Join the notes together.
  52. let result = [...(newNotes || []), ...(oldNotes || [])]
  53. // console.log(result)
  54. log.verbose(result)
  55. resolve(result)
  56. })
  57. }
  58. function getNewNotesiOS9 (backup) {
  59. return new Promise((resolve, reject) => {
  60. backup.openDatabase(NOTES2_DB)
  61. .then(db => {
  62. db.all(`SELECT ZICCLOUDSYNCINGOBJECT.*, ZICNOTEDATA.ZDATA as X_CONTENT_DATA, datetime(ZCREATIONDATE + 978307200, 'unixepoch') AS XFORMATTEDDATESTRING FROM ZICCLOUDSYNCINGOBJECT LEFT JOIN ZICNOTEDATA ON ZICCLOUDSYNCINGOBJECT.ZNOTE = ZICNOTEDATA.ZNOTE`, async function (err, rows) {
  63. if (err) reject(err)
  64. resolve(rows)
  65. })
  66. })
  67. .catch(reject)
  68. })
  69. }
  70. function getOldNotes (backup) {
  71. return new Promise((resolve, reject) => {
  72. backup.openDatabase(NOTES_DB)
  73. .then(db => {
  74. db.all(`SELECT *, datetime(ZCREATIONDATE + 978307200, 'unixepoch') AS XFORMATTEDDATESTRING from ZNOTE LEFT JOIN ZNOTEBODY ON ZBODY = ZNOTEBODY.Z_PK`, function (err, rows) {
  75. if (err) reject(err)
  76. resolve(rows)
  77. })
  78. })
  79. .catch(reject)
  80. })
  81. }
  82. function getNewNotesiOS10iOS11 (backup) {
  83. return new Promise((resolve, reject) => {
  84. backup.openDatabase(NOTES2_DB)
  85. .then(db => {
  86. db.all(`SELECT ZICCLOUDSYNCINGOBJECT.*, ZICNOTEDATA.ZDATA as X_CONTENT_DATA, datetime(ZCREATIONDATE + 978307200, 'unixepoch') AS XFORMATTEDDATESTRING, datetime(ZCREATIONDATE1 + 978307200, 'unixepoch') AS XFORMATTEDDATESTRING1 FROM ZICCLOUDSYNCINGOBJECT LEFT JOIN ZICNOTEDATA ON ZICCLOUDSYNCINGOBJECT.ZNOTE = ZICNOTEDATA.ZNOTE`, function (err, rows) {
  87. if (err) reject(err)
  88. resolve(rows)
  89. })
  90. })
  91. .catch(reject)
  92. })
  93. }