web-dev-qa-db-ja.com

React.js、node.js、webpack、babel、expressでfsモジュールを使用します

フォームを表示するビューをレンダリングするという要件があります。フォームの送信時に、フォームデータを収集してファイルを作成し、フォームデータをJSONとしてそのファイルに保存する必要があります。 React.js、node.js、babel、webpackを使用しています。

これを達成するために少し苦労した後、私は同形またはユニバーサルJavaScriptを使用する必要があることを理解しました。つまり、クライアント側でfsモジュールを使用できないため、サーバー側で反応およびレンダリングを使用する必要があります。 サーバー側についてはこれを参照

私はそれを使用して実行します:npm run start

この後、コンソールで[Object Object]は、以下の反応コンポーネント(HomePage.js)の1行目からコンソールに出力されます。しかし、後でこのページにアクセスすると、エラーが発生します。

'bundle.js:18キャッチされないエラー:モジュール "fs"が見つかりません'

reactでfsモジュールをどのように使用できますか?

以下はコードスニペットです。

webpack.config.js

"use strict";

const debug = process.env.NODE_ENV !== "production";

const webpack = require('webpack');
const path = require('path');

module.exports = {
  devtool: debug ? 'inline-sourcemap' : null,
  entry: path.join(__dirname, 'src', 'app-client.js'),
  devServer: {
    inline: true,
    port: 3333,
    contentBase: "src/static/",
    historyApiFallback: true
  },
  output: {
    path: path.join(__dirname, 'src', 'static', 'js'),
    publicPath: "/js/",
    filename: 'bundle.js'
  },
  module: {
    loaders: [{
      test: path.join(__dirname, 'src'),
      loader: ['babel-loader'],
      query: {
        //cacheDirectory: 'babel_cache',
        presets: debug ? ['react', 'es2015', 'react-hmre'] : ['react', 'es2015']
      }
    }]
  },
  plugins: debug ? [] : [
    new webpack.DefinePlugin({
      'process.env.NODE_ENV': JSON.stringify(process.env.NODE_ENV)
    }),
    new webpack.optimize.DedupePlugin(),
    new webpack.optimize.OccurenceOrderPlugin(),
    new webpack.optimize.UglifyJsPlugin({
      compress: { warnings: false },
      mangle: true,
      sourcemap: false,
      beautify: false,
      dead_code: true
    }),
  ]
};

package.json

{
  "name": "sample",
  "version": "1.0.0",
  "description": "Simple application to showcase how to achieve universal rendering and routing with React and Express.",
  "main": "src/server.js",
  "scripts": {
    "start": "SET NODE_ENV=production&&babel-node src/server.js",
    "start-dev": "npm run start-dev-hmr",
    "start-dev-single-page": "node_modules/.bin/http-server src/static",
    "start-dev-hmr": "webpack-dev-server --progress --inline --hot",
    "build": "SET NODE_ENV=production&&webpack -p"
  },
  "dependencies": {
    "babel-cli": "^6.11.4",
    "babel-core": "^6.13.2",
    "babel-loader": "^6.2.5",
    "babel-plugin-react-html-attrs": "^2.0.0",
    "babel-preset-es2015": "^6.13.2",
    "babel-preset-react": "^6.11.1",
    "babel-preset-react-hmre": "^1.1.1",
    "ejs": "^2.5.1",
    "express": "^4.14.0",
    "react": "^15.3.1",
    "react-dom": "^15.3.1",
    "react-router": "^2.6.1"
  },
  "devDependencies": {
    "http-server": "^0.9.0",
    "react-hot-loader": "^1.3.0",
    "webpack": "^1.13.2",
    "webpack-dev-server": "^1.14.1"
  }
}

server.js

use strict';

import path from 'path';
import { Server } from 'http';
import Express from 'express';
import React from 'react';
import { renderToString } from 'react-dom/server';
import { match, RouterContext } from 'react-router';
import routes from './routes';
import NotFoundPage from './components/NotFoundPage';
//import fs from 'fs';

//console.log("server" + fs);
// initialize the server and configure support for ejs templates
const app = new Express();
const server = new Server(app);
app.set('view engine', 'ejs');
app.set('views', path.join(__dirname, 'views'));

// define the folder that will be used for static assets
app.use(Express.static(path.join(__dirname, 'static')));

// universal routing and rendering
app.get('*', (req, res) => {
  match(
    { routes, location: req.url },
    (err, redirectLocation, renderProps) => {
//console.log("renderProps "+ Object.values(routes));
//console.log("req.url "+ req.url);
      // in case of error display the error message
      if (err) {
        return res.status(500).send(err.message);
      }

      // in case of redirect propagate the redirect to the browser
      if (redirectLocation) {
        return res.redirect(302, redirectLocation.pathname + redirectLocation.search);
      }

      // generate the React markup for the current route
      let markup;
      if (renderProps) {
        // if the current route matched we have renderProps
        markup = renderToString(<RouterContext {...renderProps}/>);
      } else {
        // otherwise we can render a 404 page
        markup = renderToString(<NotFoundPage/>);
        res.status(404);
      }

      // render the index template with the embedded React markup
      return res.render('index', { markup });
    }
  );
});

// start the server
const port = process.env.PORT || 3000;
const env = process.env.NODE_ENV || 'production';
console.log(`Server starting on http://localhost:${port} [${env}]`)
server.listen(port, err => {
  if (err) {
    return console.error(err);
  }
  console.info(`Server running on http://localhost:${port} [${env}]`);
});

HomePage.js(Reactコンポーネント)

import React from 'react';
import fs from 'fs';  
import dateformat from 'dateformat';
console.log("home page" + fs);  -- Line 1
class HomePage extends React.Component{
 checkDirectory(directory, callback) {
    fs.stat(directory, function(err, stats) {
      //Check if error defined and the error code is "not exists"
      if (err && err.errno === 34) {
        //Create the directory, call the callback.
        fs.mkdir(directory, callback);
      } else {
        //just in case there was a different error:
        callback(err)
      }
    });
  }
 handleClick(){


    var obj = JSON.stringify($('#statusForm').serializeArray());
    
    this.checkDirectory("directory/"+currentDate, function(error) {
      if(error) {
        console.log("oh no!!!", error);
      } else {
        //Carry on, all good, directory exists / created.
        fs.writeFile("directory/"+currentDate+name+".json", obj, function(err) {
        if(err) {
            return console.log(err);
        }

        console.log("The file was saved!");
        });
        console.log("exists");
      }
    });*/

  }
  render() {
    return (
      <div className="container">

    <form id="statusForm" className="form-horizontal" >
      <div className="form-group">
        <label className="control-label col-sm-2" for="names">Select list:</label>
        <div className="col-sm-10">
          <select name="names" className="form-control" id="names">
            <option>Select</option>
            <option>abc</option>
            <option>xyz</option>
          </select>
        </div>
      </div>
      <div className="form-group">
        <label className="control-label col-sm-2" for="team">Select list:</label>
        <div className="col-sm-10">
          <select name="team" className="form-control" id="team">
            <option>Select</option>
            <option>team 1</option>
            <option>team 2</option>
          </select>
        </div>
      </div>
      <div className="form-group">
        <label className="control-label col-sm-2" for="pwd">Password:</label>
        <div className="col-sm-10">
          <input type="textarea" className="form-control" id="todayTask" name="todayTask" placeholder="Enter Task"/>
        </div>
      </div>
      <div className="form-group">
        <div className="col-sm-offset-2 col-sm-10">
          <button type="button" className="btn btn-default" onClick={this.handleClick.bind(this)}>Submit</button>
        </div>
      </div>
    </form>
  </div>
    );
  }
}


export default HomePage;

EDIT 1:

さらに調査したところ、npm run buildを使用してアプリを明示的にビルドせず、reactコンポーネントを更新するだけでは、上記のエラーが発生しないことがわかりました。また、この後、ファイル作成ロジックをrenderメソッド内に直接配置し、ページを更新すると、ファイルが正常に作成されます。したがって、ボタンのOnclickでは機能せず、ページを更新すると機能する可能性があります。それはサーバーに行き、それがこのように機能する理由です。

編集2:

Webpack構成でtarget: 'node'を使用することでページ更新の問題が解決されましたが、エラーが発生します:

Uncaught ReferenceError:requireが定義されていません

Browser.soでは、renderメソッド内に直接ファイル作成ロジックがあり、ページにアクセスした瞬間にファイルが作成されます。更新は必要ありません。

希望の要件を達成するための最良の方法は何ですか?

8
SCoder

エラー

まず、エラーを少し見てみましょう。

npm run buildまたはnpm run startを使用しない場合は、webpackを使用しないため、requireステートメントがfsモジュールの内容に置き換えられず、代わりにrequireが残りますrequireはノードのみの関数であるため、ブラウザが理解できないステートメント。したがって、エラーは定義する必要はありません。

npm run buildまたはnpm run startで実行する場合、webpackはそのrequireステートメントを取り出し、それをfsモジュールに置き換えます。しかし、あなたが発見したように、fsはクライアント側では機能しません。

代替案

では、fsを使用してファイルを保存できない場合は、どうすればよいでしょうか。

ファイルをサーバーに保存しようとしている場合は、フォームからNodeサーバーにデータを送信する必要があり、Nodeサーバーはfsを使用できますサーバーのファイルシステムと対話してファイルを保存します。

フォームをローカルに保存しようとしている場合、つまりブラウザと同じデバイスに保存しようとしている場合は、 this のような別の戦略を使用するか、 FileSaver のようなクライアント側ライブラリ。どちらのオプションを選択するかはユースケースによって多少異なりますが、クライアント側で保存しようとしている場合は、「Webブラウザーからファイルを保存する」または「クライアント側でファイルを保存する」を検索して、何が効果的かを確認できます。

7
hellojeffhall