index.js 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. /* jshint node: true */
  2. "use strict";
  3. /**
  4. * A module to get Android versions by API level, NDK level, semantic version, or version name.
  5. *
  6. * Versions are referenced from here:
  7. * {@link https://source.android.com/source/build-numbers.html#platform-code-names-versions-api-levels-and-ndk-releases}
  8. * {@link https://github.com/android/platform_frameworks_base/blob/master/core/java/android/os/Build.java}
  9. *
  10. * The version for "Current Development Build" ("CUR_DEVELOPMENT") is not included.
  11. *
  12. * @module android-versions
  13. */
  14. var VERSIONS = {
  15. BASE: { api: 1, ndk: 0, semver: "1.0", name: "(no code name)", },
  16. BASE_1_1: { api: 2, ndk: 0, semver: "1.1", name: "(no code name)", },
  17. CUPCAKE: { api: 3, ndk: 1, semver: "1.5", name: "Cupcake", },
  18. DONUT: { api: 4, ndk: 2, semver: "1.6", name: "Donut", },
  19. ECLAIR: { api: 5, ndk: 2, semver: "2.0", name: "Eclair", },
  20. ECLAIR_0_1: { api: 6, ndk: 2, semver: "2.0.1", name: "Eclair", },
  21. ECLAIR_MR1: { api: 7, ndk: 3, semver: "2.1", name: "Eclair", },
  22. FROYO: { api: 8, ndk: 4, semver: "2.2.x", name: "Froyo", },
  23. GINGERBREAD: { api: 9, ndk: 5, semver: "2.3.0 - 2.3.2", name: "Gingerbread", },
  24. GINGERBREAD_MR1: { api: 10, ndk: 5, semver: "2.3.3 - 2.3.7", name: "Gingerbread", },
  25. HONEYCOMB: { api: 11, ndk: 5, semver: "3.0", name: "Honeycomb", },
  26. HONEYCOMB_MR1: { api: 12, ndk: 6, semver: "3.1", name: "Honeycomb", },
  27. HONEYCOMB_MR2: { api: 13, ndk: 6, semver: "3.2.x", name: "Honeycomb", },
  28. ICE_CREAM_SANDWICH: { api: 14, ndk: 7, semver: "4.0.1 - 4.0.2", name: "Ice Cream Sandwich", },
  29. ICE_CREAM_SANDWICH_MR1: { api: 15, ndk: 8, semver: "4.0.3 - 4.0.4", name: "Ice Cream Sandwich", },
  30. JELLY_BEAN: { api: 16, ndk: 8, semver: "4.1.x", name: "Jellybean", },
  31. JELLY_BEAN_MR1: { api: 17, ndk: 8, semver: "4.2.x", name: "Jellybean", },
  32. JELLY_BEAN_MR2: { api: 18, ndk: 8, semver: "4.3.x", name: "Jellybean", },
  33. KITKAT: { api: 19, ndk: 8, semver: "4.4.0 - 4.4.4", name: "KitKat", },
  34. KITKAT_WATCH: { api: 20, ndk: 8, semver: "4.4", name: "KitKat Watch", },
  35. LOLLIPOP: { api: 21, ndk: 8, semver: "5.0", name: "Lollipop", },
  36. LOLLIPOP_MR1: { api: 22, ndk: 8, semver: "5.1", name: "Lollipop", },
  37. M: { api: 23, ndk: 8, semver: "6.0", name: "Marshmallow", },
  38. N: { api: 24, ndk: 8, semver: "7.0", name: "Nougat", },
  39. N_MR1: { api: 25, ndk: 8, semver: "7.1", name: "Nougat", },
  40. O: { api: 26, ndk: 8, semver: "8.0.0", name: "Oreo", },
  41. O_MR1: { api: 27, ndk: 8, semver: "8.1.0", name: "Oreo", },
  42. P: { api: 28, ndk: 8, semver: "9", name: "Pie", },
  43. Q: { api: 29, ndk: 8, semver: "10", name: "Android10", },
  44. R: { api: 30, ndk: 8, semver: "11", name: "Android11", },
  45. S: { api: 31, ndk: 8, semver: "12", name: "Android12", },
  46. S_V2: { api: 32, ndk: 8, semver: "12", name: "Android12L", },
  47. TIRAMISU: { api: 33, ndk: 8, semver: "13", name: "Android13", },
  48. UPSIDE_DOWN_CAKE: { api: 34, ndk: 8, semver: "14", name: "Android14", }
  49. }
  50. // Add a key to each version of Android for the "versionCode".
  51. // This is the same key we use in the VERSIONS map above.
  52. Object.keys(VERSIONS).forEach(function(version) {
  53. VERSIONS[version].versionCode = version
  54. })
  55. var semver = require('semver');
  56. // semver format requires <major>.<minor>.<patch> but we allow just <major>.<minor> format.
  57. // Coerce <major>.<minor> to <major>.<minor>.0
  58. function formatSemver(semver) {
  59. if (semver.match(/^\d+.\d+$/)) {
  60. return semver + '.0'
  61. } else {
  62. return semver
  63. }
  64. }
  65. // The default predicate compares against API level, semver, name, or code.
  66. function getFromDefaultPredicate(arg) {
  67. // Coerce arg to string for comparisons below.
  68. arg = arg.toString()
  69. return getFromPredicate(function(version) {
  70. // Check API level before all else.
  71. if (arg === version.api.toString()) {
  72. return true
  73. }
  74. var argSemver = formatSemver(arg)
  75. if (semver.valid(argSemver) && semver.satisfies(argSemver, version.semver)) {
  76. return true
  77. }
  78. // Compare version name and code.
  79. return arg === version.name || arg === version.versionCode
  80. })
  81. }
  82. // The function to allow passing a predicate.
  83. function getFromPredicate(predicate) {
  84. if (predicate === null) {
  85. return null
  86. }
  87. return Object.keys(VERSIONS).filter(function(version) {
  88. return predicate(VERSIONS[version])
  89. }).map(function(key) { return VERSIONS[key] })
  90. }
  91. /**
  92. * The Android version codes available as keys for easier look-up.
  93. */
  94. Object.keys(VERSIONS).forEach(function(name) {
  95. exports[name] = VERSIONS[name]
  96. })
  97. /**
  98. * The complete reference of Android versions for easier look-up.
  99. */
  100. exports.VERSIONS = VERSIONS
  101. /**
  102. * Retrieve a single Android version.
  103. *
  104. * @param {object | Function} arg - The value or predicate to use to retrieve values.
  105. *
  106. * @return {object} An object representing the version found or null if none found.
  107. */
  108. exports.get = function(arg) {
  109. var result = exports.getAll(arg)
  110. if (result === null || result.length === 0) {
  111. return null
  112. }
  113. return result[0]
  114. }
  115. /**
  116. * Retrieve all Android versions that meet the criteria of the argument.
  117. *
  118. * @param {object | Function} arg - The value or predicate to use to retrieve values.
  119. *
  120. * @return {object} An object representing the version found or null if none found.
  121. */
  122. exports.getAll = function(arg) {
  123. if (arg === null) {
  124. return null
  125. }
  126. if (typeof arg === "function") {
  127. return getFromPredicate(arg)
  128. } else {
  129. return getFromDefaultPredicate(arg)
  130. }
  131. }