1. 这是什么?

validate-npm-package-name的作用就是:检验npm包的名称是否符合命名标准。

官网中也列出这个工具应该如何使用且举例了正确的包名称

  1. var validate = require("validate-npm-package-name")
  2. validate("some-package")
  3. validate("example.com")
  4. validate("under_score")
  5. validate("123numeric")
  6. validate("@npm/thingy")
  7. validate("@jane/foo.js")

2.学习目标

粗略的看了一下源码的内容,能大致知道通过本次阅读能学习到的知识有:

  • 知道正确的包名称应该怎么命名
  • 简单理解正则表达式

3.阅读源码

整个源码分为两个函数

  • validate(),接收一个参数——包名称
  • done(),接收两个参数——警告数组、错误数组

3.1 validate()

这部分主要是一些字符串判断的校验

  1. var validate = module.exports = function (name) {
  2. var warnings = []
  3. var errors = []
  4. // 不能为null
  5. if (name === null) {
  6. errors.push('name cannot be null')
  7. return done(warnings, errors)
  8. }
  9. // 不能为undefined
  10. if (name === undefined) {
  11. errors.push('name cannot be undefined')
  12. return done(warnings, errors)
  13. }
  14. // 必须是一个字符串
  15. if (typeof name !== 'string') {
  16. errors.push('name must be a string')
  17. return done(warnings, errors)
  18. }
  19. // 名称不能为空
  20. if (!name.length) {
  21. errors.push('name length must be greater than zero')
  22. }
  23. // 名称不能以.开头
  24. if (name.match(/^\./)) {
  25. errors.push('name cannot start with a period')
  26. }
  27. // 名称不能以下划线开头
  28. if (name.match(/^_/)) {
  29. errors.push('name cannot start with an underscore')
  30. }
  31. // 名称尾部不能有空格
  32. if (name.trim() !== name) {
  33. errors.push('name cannot contain leading or trailing spaces')
  34. }
  35. /* var blacklist = [
  36. 'node_modules',
  37. 'favicon.ico'
  38. ]
  39. */
  40. // 不能出现黑名单中的名字
  41. // No funny business
  42. blacklist.forEach(function (blacklistedName) {
  43. if (name.toLowerCase() === blacklistedName) {
  44. errors.push(blacklistedName + ' is a blacklisted name')
  45. }
  46. })
  47. // 不能是node内置模块的名称,这里引入了一个包,包里是node内置模块的数组
  48. // core module names like http, events, util, etc
  49. builtins.forEach(function (builtin) {
  50. if (name.toLowerCase() === builtin) {
  51. warnings.push(builtin + ' is a core module name')
  52. }
  53. })
  54. // 名称不可以很长很长很长很长
  55. if (name.length > 214) {
  56. warnings.push('name can no longer contain more than 214 characters')
  57. }
  58. // 大小写不能同时出现
  59. if (name.toLowerCase() !== name) {
  60. warnings.push('name can no longer contain capital letters')
  61. }
  62. // 不能包含()~ ! *等符号
  63. if (/[~'!()*]/.test(name.split('/').slice(-1)[0])) {
  64. warnings.push('name can no longer contain special characters ("~\'!()*")')
  65. }
  66. // 有些内容可能会出现歧义,所以需要encode一下,比如出现/,或@baidu/anti-spam
  67. // 如果encode前后内容一致,则会通过
  68. // 否则会报错
  69. if (encodeURIComponent(name) !== name) {
  70. // Maybe it's a scoped package name, like @user/package
  71. var nameMatch = name.match(scopedPackagePattern)
  72. if (nameMatch) {
  73. var user = nameMatch[1]
  74. var pkg = nameMatch[2]
  75. if (encodeURIComponent(user) === user && encodeURIComponent(pkg) === pkg) {
  76. return done(warnings, errors)
  77. }
  78. }
  79. errors.push('name can only contain URL-friendly characters')
  80. }
  81. return done(warnings, errors)
  82. }

这一部分已经将所有包命名规则都校验了一遍,如果出现不合理的情况时,则会往errors、warnings数组中push错误信息

3.2 done()

这一部分则是对校验完的参数进行处理

  1. var done = function (warnings, errors) {
  2. var result = {
  3. // 如果没有错误和警告,就为true
  4. validForNewPackages: errors.length === 0 && warnings.length === 0,
  5. // 如果没有错误就为true
  6. validForOldPackages: errors.length === 0,
  7. warnings: warnings,
  8. errors: errors
  9. }
  10. // 检查errors、warnings数组的长度,如果有空,则删除该属性
  11. if (!result.warnings.length) delete result.warnings
  12. if (!result.errors.length) delete result.errors
  13. // 返回一个校验后的最终结果对象
  14. return result
  15. }

4. 总结

这次的源码较其他期比较简单,但也能学习到如果需要制定一个公共的约定时,需要考虑的判断因素会比较多,各方面都需要进行考虑。因此在项目开发中,对复杂的业务场景进行处理时,也应该考虑各种情况的出现,避免代码出现问题