summaryrefslogtreecommitdiff
path: root/deps/npm/lib/profile.js
blob: 24f026ce85ec48a0572635a9d680347ae2bda936 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
const inspect = require('util').inspect
const { URL } = require('url')
const ansistyles = require('ansistyles')
const log = require('npmlog')
const npmProfile = require('npm-profile')
const qrcodeTerminal = require('qrcode-terminal')
const Table = require('cli-table3')

const npm = require('./npm.js')
const otplease = require('./utils/otplease.js')
const output = require('./utils/output.js')
const pulseTillDone = require('./utils/pulse-till-done.js')
const readUserInfo = require('./utils/read-user-info.js')
const usageUtil = require('./utils/usage.js')

const usage = usageUtil(
  'npm profile enable-2fa [auth-only|auth-and-writes]\n',
  'npm profile disable-2fa\n',
  'npm profile get [<key>]\n',
  'npm profile set <key> <value>'
)

const completion = (opts, cb) => {
  var argv = opts.conf.argv.remain
  const subcommands = ['enable-2fa', 'disable-2fa', 'get', 'set']

  if (!argv[2])
    return cb(null, subcommands)

  switch (argv[2]) {
    case 'enable-2fa':
    case 'enable-tfa':
      return cb(null, ['auth-and-writes', 'auth-only'])

    case 'disable-2fa':
    case 'disable-tfa':
    case 'get':
    case 'set':
      return cb(null, [])
    default:
      return cb(new Error(argv[2] + ' not recognized'))
  }
}

const cmd = (args, cb) => profile(args).then(() => cb()).catch(cb)

const profile = async (args) => {
  if (args.length === 0)
    throw new Error(usage)

  log.gauge.show('profile')

  const [subcmd, ...opts] = args

  switch (subcmd) {
    case 'enable-2fa':
    case 'enable-tfa':
    case 'enable2fa':
    case 'enabletfa':
      return enable2fa(opts)
    case 'disable-2fa':
    case 'disable-tfa':
    case 'disable2fa':
    case 'disabletfa':
      return disable2fa()
    case 'get':
      return get(opts)
    case 'set':
      return set(opts)
    default:
      throw new Error('Unknown profile command: ' + subcmd)
  }
}

const knownProfileKeys = [
  'name',
  'email',
  'two-factor auth',
  'fullname',
  'homepage',
  'freenode',
  'twitter',
  'github',
  'created',
  'updated',
]

const get = async args => {
  const tfa = 'two-factor auth'
  const conf = { ...npm.flatOptions }

  const info = await pulseTillDone.withPromise(npmProfile.get(conf))

  if (!info.cidr_whitelist)
    delete info.cidr_whitelist

  if (conf.json) {
    output(JSON.stringify(info, null, 2))
    return
  }

  // clean up and format key/values for output
  const cleaned = {}
  for (const key of knownProfileKeys)
    cleaned[key] = info[key] || ''

  const unknownProfileKeys = Object.keys(info).filter((k) => !(k in cleaned))
  for (const key of unknownProfileKeys)
    cleaned[key] = info[key] || ''

  delete cleaned.tfa
  delete cleaned.email_verified
  cleaned.email += info.email_verified ? ' (verified)' : '(unverified)'

  if (info.tfa && !info.tfa.pending)
    cleaned[tfa] = info.tfa.mode
  else
    cleaned[tfa] = 'disabled'

  if (args.length) {
    const values = args // comma or space separated
      .join(',')
      .split(/,/)
      .filter((arg) => arg.trim() !== '')
      .map((arg) => cleaned[arg])
      .join('\t')
    output(values)
  } else {
    if (conf.parseable) {
      for (const key of Object.keys(info)) {
        if (key === 'tfa')
          output(`${key}\t${cleaned[tfa]}`)
        else
          output(`${key}\t${info[key]}`)
      }
    } else {
      const table = new Table()
      for (const key of Object.keys(cleaned))
        table.push({ [ansistyles.bright(key)]: cleaned[key] })

      output(table.toString())
    }
  }
}

const writableProfileKeys = [
  'email',
  'password',
  'fullname',
  'homepage',
  'freenode',
  'twitter',
  'github',
]

const set = async (args) => {
  const conf = { ...npm.flatOptions }
  const prop = (args[0] || '').toLowerCase().trim()

  let value = args.length > 1 ? args.slice(1).join(' ') : null

  const readPasswords = async () => {
    const newpassword = await readUserInfo.password('New password: ')
    const confirmedpassword = await readUserInfo.password('       Again:     ')

    if (newpassword !== confirmedpassword) {
      log.warn('profile', 'Passwords do not match, please try again.')
      return readPasswords()
    }

    return newpassword
  }

  if (prop !== 'password' && value === null)
    throw new Error('npm profile set <prop> <value>')

  if (prop === 'password' && value !== null) {
    throw new Error(
      'npm profile set password\n' +
      'Do not include your current or new passwords on the command line.')
  }

  if (writableProfileKeys.indexOf(prop) === -1) {
    throw new Error(`"${prop}" is not a property we can set. ` +
      `Valid properties are: ` + writableProfileKeys.join(', '))
  }

  if (prop === 'password') {
    const current = await readUserInfo.password('Current password: ')
    const newpassword = await readPasswords()

    value = { old: current, new: newpassword }
  }

  // FIXME: Work around to not clear everything other than what we're setting
  const user = await pulseTillDone.withPromise(npmProfile.get(conf))
  const newUser = {}

  for (const key of writableProfileKeys)
    newUser[key] = user[key]

  newUser[prop] = value

  const result = await otplease(conf, conf => npmProfile.set(newUser, conf))

  if (conf.json)
    output(JSON.stringify({ [prop]: result[prop] }, null, 2))
  else if (conf.parseable)
    output(prop + '\t' + result[prop])
  else if (result[prop] != null)
    output('Set', prop, 'to', result[prop])
  else
    output('Set', prop)
}

const enable2fa = async (args) => {
  if (args.length > 1)
    throw new Error('npm profile enable-2fa [auth-and-writes|auth-only]')

  const mode = args[0] || 'auth-and-writes'
  if (mode !== 'auth-only' && mode !== 'auth-and-writes') {
    throw new Error(
      `Invalid two-factor authentication mode "${mode}".\n` +
      'Valid modes are:\n' +
      '  auth-only - Require two-factor authentication only when logging in\n' +
      '  auth-and-writes - Require two-factor authentication when logging in ' +
      'AND when publishing'
    )
  }

  const conf = { ...npm.flatOptions }
  if (conf.json || conf.parseable) {
    throw new Error(
      'Enabling two-factor authentication is an interactive operation and ' +
      (conf.json ? 'JSON' : 'parseable') + ' output mode is not available'
    )
  }

  const info = {
    tfa: {
      mode: mode,
    },
  }

  // if they're using legacy auth currently then we have to
  // update them to a bearer token before continuing.
  const auth = getAuth(conf)

  if (!auth.basic && !auth.token) {
    throw new Error(
      'You need to be logged in to registry ' +
      `${conf.registry} in order to enable 2fa`
    )
  }

  if (auth.basic) {
    log.info('profile', 'Updating authentication to bearer token')
    const result = await npmProfile.createToken(
      auth.basic.password, false, [], conf
    )

    if (!result.token) {
      throw new Error(
        `Your registry ${conf.registry} does not seem to ` +
        'support bearer tokens. Bearer tokens are required for ' +
        'two-factor authentication'
      )
    }

    npm.config.setCredentialsByURI(conf.registry, { token: result.token })
    await npm.config.save('user')
  }

  log.notice('profile', 'Enabling two factor authentication for ' + mode)
  const password = await readUserInfo.password()
  info.tfa.password = password

  log.info('profile', 'Determine if tfa is pending')
  const userInfo = await pulseTillDone.withPromise(npmProfile.get(conf))

  if (userInfo && userInfo.tfa && userInfo.tfa.pending) {
    log.info('profile', 'Resetting two-factor authentication')
    await pulseTillDone.withPromise(
      npmProfile.set({ tfa: { password, mode: 'disable' } }, conf)
    )
  } else if (userInfo && userInfo.tfa) {
    if (conf.otp)
      conf.otp = conf.otp
    else {
      const otp = await readUserInfo.otp(
        'Enter one-time password from your authenticator app: '
      )
      conf.otp = otp
    }
  }

  log.info('profile', 'Setting two-factor authentication to ' + mode)
  const challenge = await pulseTillDone.withPromise(npmProfile.set(info, conf))

  if (challenge.tfa === null) {
    output('Two factor authentication mode changed to: ' + mode)
    return
  }

  const badResponse = typeof challenge.tfa !== 'string'
    || !/^otpauth:[/][/]/.test(challenge.tfa)
  if (badResponse) {
    throw new Error(
      'Unknown error enabling two-factor authentication. Expected otpauth URL' +
      ', got: ' + inspect(challenge.tfa)
    )
  }

  const otpauth = new URL(challenge.tfa)
  const secret = otpauth.searchParams.get('secret')
  const code = await qrcode(challenge.tfa)

  output(
    'Scan into your authenticator app:\n' + code + '\n Or enter code:', secret
  )

  const interactiveOTP =
    await readUserInfo.otp('And an OTP code from your authenticator: ')

  log.info('profile', 'Finalizing two-factor authentication')

  const result = await npmProfile.set({ tfa: [interactiveOTP] }, conf)

  output(
    '2FA successfully enabled. Below are your recovery codes, ' +
    'please print these out.'
  )
  output(
    'You will need these to recover access to your account ' +
    'if you lose your authentication device.'
  )

  for (const tfaCode of result.tfa)
    output('\t' + tfaCode)
}

const getAuth = conf => {
  const creds = npm.config.getCredentialsByURI(conf.registry)
  const auth = {}

  if (creds.token)
    auth.token = creds.token
  else if (creds.username)
    auth.basic = { username: creds.username, password: creds.password }
  else if (creds.auth) {
    const basic = Buffer.from(creds.auth, 'base64').toString().split(':', 2)
    auth.basic = { username: basic[0], password: basic[1] }
  }

  if (conf.otp)
    auth.otp = conf.otp

  return auth
}

const disable2fa = async args => {
  const conf = { ...npm.flatOptions }
  const info = await pulseTillDone.withPromise(npmProfile.get(conf))

  if (!info.tfa || info.tfa.pending) {
    output('Two factor authentication not enabled.')
    return
  }

  const password = await readUserInfo.password()

  if (!conf.otp) {
    const msg = 'Enter one-time password from your authenticator app: '
    conf.otp = await readUserInfo.otp(msg)
  }

  log.info('profile', 'disabling tfa')

  await pulseTillDone.withPromise(npmProfile.set({
    tfa: { password: password, mode: 'disable' },
  }, conf))

  if (conf.json)
    output(JSON.stringify({ tfa: false }, null, 2))
  else if (conf.parseable)
    output('tfa\tfalse')
  else
    output('Two factor authentication disabled.')
}

const qrcode = url =>
  new Promise((resolve) => qrcodeTerminal.generate(url, resolve))

module.exports = Object.assign(cmd, { usage, completion })