123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147 |
- 'use strict';
- const _ = {
- isPlainObject: require('lodash/isPlainObject'),
- get: require('lodash/get'),
- set: require('lodash/set'),
- };
- const { defer, empty, from, of } = require('rxjs');
- const { concatMap, filter, publish, reduce } = require('rxjs/operators');
- const runAsync = require('run-async');
- const utils = require('../utils/utils');
- const Base = require('./baseUI');
- /**
- * Base interface class other can inherits from
- */
- class PromptUI extends Base {
- constructor(prompts, opt) {
- super(opt);
- this.prompts = prompts;
- }
- run(questions, answers) {
- // Keep global reference to the answers
- if (_.isPlainObject(answers)) {
- this.answers = { ...answers };
- } else {
- this.answers = {};
- }
- // Make sure questions is an array.
- if (_.isPlainObject(questions)) {
- // It's either an object of questions or a single question
- questions = Object.values(questions).every(
- (v) => _.isPlainObject(v) && v.name === undefined
- )
- ? Object.entries(questions).map(([name, question]) => ({ name, ...question }))
- : [questions];
- }
- // Create an observable, unless we received one as parameter.
- // Note: As this is a public interface, we cannot do an instanceof check as we won't
- // be using the exact same object in memory.
- const obs = Array.isArray(questions) ? from(questions) : questions;
- this.process = obs.pipe(
- concatMap(this.processQuestion.bind(this)),
- publish() // Creates a hot Observable. It prevents duplicating prompts.
- );
- this.process.connect();
- return this.process
- .pipe(
- reduce((answers, answer) => {
- _.set(answers, answer.name, answer.answer);
- return answers;
- }, this.answers)
- )
- .toPromise(Promise)
- .then(this.onCompletion.bind(this), this.onError.bind(this));
- }
- /**
- * Once all prompt are over
- */
- onCompletion() {
- this.close();
- return this.answers;
- }
- onError(error) {
- this.close();
- return Promise.reject(error);
- }
- processQuestion(question) {
- question = { ...question };
- return defer(() => {
- const obs = of(question);
- return obs.pipe(
- concatMap(this.setDefaultType.bind(this)),
- concatMap(this.filterIfRunnable.bind(this)),
- concatMap(() =>
- utils.fetchAsyncQuestionProperty(question, 'message', this.answers)
- ),
- concatMap(() =>
- utils.fetchAsyncQuestionProperty(question, 'default', this.answers)
- ),
- concatMap(() =>
- utils.fetchAsyncQuestionProperty(question, 'choices', this.answers)
- ),
- concatMap(this.fetchAnswer.bind(this))
- );
- });
- }
- fetchAnswer(question) {
- const Prompt = this.prompts[question.type];
- this.activePrompt = new Prompt(question, this.rl, this.answers);
- return defer(() =>
- from(this.activePrompt.run().then((answer) => ({ name: question.name, answer })))
- );
- }
- setDefaultType(question) {
- // Default type to input
- if (!this.prompts[question.type]) {
- question.type = 'input';
- }
- return defer(() => of(question));
- }
- filterIfRunnable(question) {
- if (
- question.askAnswered !== true &&
- _.get(this.answers, question.name) !== undefined
- ) {
- return empty();
- }
- if (question.when === false) {
- return empty();
- }
- if (typeof question.when !== 'function') {
- return of(question);
- }
- const { answers } = this;
- return defer(() =>
- from(
- runAsync(question.when)(answers).then((shouldRun) => {
- if (shouldRun) {
- return question;
- }
- })
- ).pipe(filter((val) => val != null))
- );
- }
- }
- module.exports = PromptUI;
|