「ハッキング・ラボのつくりかた 完全版 仮想環境におけるハッカー体験学習」と「体系的に学ぶ 安全なWebアプリケーションの作り方 第2版 脆弱性が生まれる原理と対策の実践」(通称:徳丸本)を参考に、セキュリティの勉強を進めています。
その上で必要な知識として、今回は、よく使われている Webアプリケーションのフレームワークである Vue.js を理解しています。
前回 は、Vue 2.x と Vue 3.x について、環境構築、プロジェクト作成、開発用サーバで動作確認、ビルド、ビルド後の生成物で動作確認まで行いました。
今回は、ファイル構成、各種ソースコードの内容の確認、ビルド後の生成物の内容を確認していきたいと思います。
それでは、やっていきます。
参考文献
はじめに
「セキュリティ」の記事一覧です。良かったら参考にしてください。
セキュリティの記事一覧
Vue.js とは、Webアプリケーションを構築するときに、特に UI(ユーザインタフェース)を得意としたフレームワークです。
Vue.js の公式サイトは以下です。日本語で書かれています。
ja.vuejs.org
Vue CLI の公式サイトは以下です。
cli.vuejs.org
使用する環境は、VirtualBox+Ubuntu 22.04 です。
各種ツールのバージョンです。
$ node -v
v20.13.1
$ npm -v
10.8.0
$ vue -V
@vue/cli 5.0.8
Vue 2.x の構成とビルド前のソースコード
前回作成した helloプロジェクトのビルド後の構成です。
ざっくり言うと、src 以下のソースコードを変更、追加して、public と src 以下を対象としてビルドすると、dist 以下が生成されて、最終的には dist 以下の生成物を使用します。
node_modules は、ライブラリ(パッケージ、モジュール)のようなもので、追加でインストールすると、ここに追加されていきます。かなりのファイル数ですが、中を見ることは、ほとんどないと思います。
トップにある各ファイルは、設定ファイルなどです。
tree -L 3
.
|-- README.md
|-- babel.config.js
|-- dist
| |-- css
| | `-- app.2cf79ad6.css
| |-- favicon.ico
| |-- index.html
| `-- js
| |-- app.35ff291d.js
| |-- app.35ff291d.js.map
| |-- chunk-vendors.27e771c3.js
| `-- chunk-vendors.27e771c3.js.map
|-- jsconfig.json
|-- node_modules
| |-- (省略)
|-- package-lock.json
|-- package.json
|-- public
| |-- favicon.ico
| `-- index.html
|-- src
| |-- App.vue
| |-- assets
| | `-- logo.png
| |-- components
| | `-- HelloWorld.vue
| `-- main.js
`-- vue.config.js
1328 directories, 3505 files
前回は、public/index.html の内容だけ確認しました。<div id="app"></div>
のところが、Vue.js で作ったものと置き換えられるところです。
src/main.js
src/main.js を見てみます。
vue モジュールから Vue をインポート、./App.vue から App をインポートしています。
Vue.config.productionTip = false は、true にすると、Webサーバで起動時に、ヒントが出力されるようです。
最後のところは、Vue のインスタンスを生成しています。#app が、public/index.html の id="app" を指定しているところです。
import Vue from 'vue'
import App from './App.vue'
Vue.config.productionTip = false
new Vue({
render: h => h(App),
}).$mount('#app')
src/App.vue
インポートしている vue はフレームワークの中にありますので、./App.vue を見ていきます。
<template>
内は、index.html に反映される HTML の記述で、<script>
内は JavaScript の記述で、<style>
は CSS の記述です。1つのファイルに HTML に必要なものが全て入ってるところが Vue.js の特徴のようです。見やすいですし、管理しやすいですね。
HTML に、HelloWorld というコンポーネントタグが使われています。
JavaScript に、HelloWorld コンポーネントタグの実体を読み込んでいる箇所があります。
<template>
<div id="app">
<img alt="Vue logo" src="./assets/logo.png">
<HelloWorld msg="Welcome to Your Vue.js App"/>
</div>
</template>
<script>
import HelloWorld from './components/HelloWorld.vue'
export default {
name: 'App',
components: {
HelloWorld
}
}
</script>
<style>
#app {
font-family: Avenir, Helvetica, Arial, sans-serif;
font-smoothing: antialiased;
osx-font-smoothing: grayscale;
text-align: center;
color: #2c3e50;
margin-top: 60px;
}
</style>
src/components/HelloWorld.vue
続いて、HelloWorld.vue です。
少し長いですが、HTML部分はブラウザに表示されていた内容で、{{ msg }}
以外は普通の HTML です。
JavaScript では、上の {{ msg }}
に反映するための文字列(HelloWorld)を作っています。
<template>
<div class="hello">
<h1>{{ msg }}</h1>
<p>
For a guide and recipes on how to configure / customize this project,<br>
check out the
<a href="https://cli.vuejs.org" target="_blank" rel="noopener">vue-cli documentation</a>.
</p>
<h3>Installed CLI Plugins</h3>
<ul>
<li><a href="https://github.com/vuejs/vue-cli/tree/dev/packages/%40vue/cli-plugin-babel" target="_blank" rel="noopener">babel</a></li>
<li><a href="https://github.com/vuejs/vue-cli/tree/dev/packages/%40vue/cli-plugin-eslint" target="_blank" rel="noopener">eslint</a></li>
</ul>
<h3>Essential Links</h3>
<ul>
<li><a href="https://vuejs.org" target="_blank" rel="noopener">Core Docs</a></li>
<li><a href="https://forum.vuejs.org" target="_blank" rel="noopener">Forum</a></li>
<li><a href="https://chat.vuejs.org" target="_blank" rel="noopener">Community Chat</a></li>
<li><a href="https://twitter.com/vuejs" target="_blank" rel="noopener">Twitter</a></li>
<li><a href="https://news.vuejs.org" target="_blank" rel="noopener">News</a></li>
</ul>
<h3>Ecosystem</h3>
<ul>
<li><a href="https://router.vuejs.org" target="_blank" rel="noopener">vue-router</a></li>
<li><a href="https://vuex.vuejs.org" target="_blank" rel="noopener">vuex</a></li>
<li><a href="https://github.com/vuejs/vue-devtools#vue-devtools" target="_blank" rel="noopener">vue-devtools</a></li>
<li><a href="https://vue-loader.vuejs.org" target="_blank" rel="noopener">vue-loader</a></li>
<li><a href="https://github.com/vuejs/awesome-vue" target="_blank" rel="noopener">awesome-vue</a></li>
</ul>
</div>
</template>
<script>
export default {
name: 'HelloWorld',
props: {
msg: String
}
}
</script>
<style scoped>
h3 {
margin: 40px 0 0;
}
ul {
list-style-type: none;
padding: 0;
}
li {
display: inline-block;
margin: 0 10px;
}
a {
color: #42b983;
}
</style>
こういうソースコードをベースにして、フロントエンド開発では、Vue.js の各種パーツを追加して、見栄えのいい Webアプリケーションを作っていくんだと思います。
Vue 2.x ビルド後のソースコード
実際に、Vue.js で作られた Webアプリケーションが使われるときは、先ほどの xxx.vue のファイルではなく、ビルド後のソースコードが使われます。
よって、Webアプリケーションをハッキングするときは、このビルド後のソースコードを見ることになります。
distディレクトリには 7ファイルありますが、index.html、favicon.ico、css は割愛します。
js ディレクトリには、2つの JavaScriptファイルと、対応する 2つの mapファイルがあります。この mapファイルとは、SourceMap と言って、ブラウザでデバッグするときに必要になるものらしいです。
以下は、この SourceMap を jsディレクトリに配置したまま、Webサーバを起動したときのキャプチャです。distディレクトリには main.js などは無いはずなのに、ソースコードが見えてますね。
一方、SourceMap を jsディレクトリから削除した場合の同じ状況のキャプチャです。これが本番環境の状況だと思います。2つの JavaScriptファイルのみです。
app.6a7bde24.js
2つの JavaScriptファイルのうち、この app.xxx.js は、ユーザが作成した xxx.vue から構築された JavaScript だと思います。
もう1つの chunk-vendors.xxx.js は、Vue.js のライブラリから構築されたものだと思います。
これらをエディタで開くと、圧縮というか、空白、改行が削除された状態なので、とても読みにくいです。なので、VSCode を使って、ソースコードを整形します。
拡張機能の Prettier というツールを入れると、ソースコードを全選択→右クリック→ドキュメントのフォーマットをクリックすると、読みやすく整形してくれます。
整形したソースコードと言っても、普通に見たら、全く分かりません(笑)。chunk-vendors.xxx.js は、絶対無理な感じなので、app.xxx.js を諦めたら終了です。
ブラウザの開発ツールでデバッグできるので、無理やり読んでみようと思います。ソースコードをここに貼りにくいので、記事の最後に貼ります。
まず、全体の構成を把握します。
最初から理解できませんが、今使ってる参考書の「改訂3版JavaScript本格入門 ~モダンスタイルによる基礎から現場での応用まで」によると、app.6a7bde24.js は、全体で1つの即時関数で出来ているということだと思います。
末尾に、();
があるので、定義した関数を実行しているということだと思います。
実行される順番は、まず、5個の即時関数が並んだところが実行されて、var n
のところ、n =
のところ、という順番だと思います。
5個の即時関数のところは、おそらく、事前準備のような感じだと思います。完全に予想ですけど、デバッガでステップ実行してても、あまり、意味のあるようなことはしてなかったので。
(function () {
"use strict";
var t = {
723(t, e, r){
(省略)
},
153(t){
(省略)
},
},
e = {};
function r(n) {
(省略)
}
(r.m = t), (function() {
(省略)
}(), (function() {
(省略)
}(), (function() {
(省略)
}(), (function() {
(省略)
}(), (function() {
(省略)
})();
var n = r.O(void 0, [504], function() {
return r(723);
});
n = r.O(n);
})();
デバッガでしばらくステップ実行してましたが、今の JavaScript の文法の理解度では、意味を読み取るのは難しかったです。もう少し文法を理解してから再挑戦したいと思います。
おわりに
今回は、Vue.js のソースコードを確認しました。JavaScript にまだ慣れてなくて、ビルド後のソースコードを読むのは難しかったです。
JavaScript は、公式のロゴが存在しないらしいという情報が見つかりました。公式ではないですが、よく見るやつを、ありがたく使わせていただきます。
最後になりましたが、エンジニアグループのランキングに参加中です。
気楽にポチッとよろしくお願いいたします🙇
今回は以上です!
最後までお読みいただき、ありがとうございました。
付録:app.6a7bde24.jsの全文
(function () {
"use strict";
var t = {
723: function (t, e, r) {
var n = r(471),
l = function () {
var t = this,
e = t._self._c;
return e(
"div",
{
attrs: {
id: "app",
},
},
[
e("img", {
attrs: {
alt: "Vue logo",
src: r(153),
},
}),
e("HelloWorld", {
attrs: {
msg: "Welcome to Your Vue.js App",
},
}),
],
1
);
},
o = [],
i = function () {
var t = this,
e = t._self._c;
return e(
"div",
{
staticClass: "hello",
},
[
e("h1", [t._v(t._s(t.msg))]),
t._m(0),
e("h3", [t._v("Installed CLI Plugins")]),
t._m(1),
e("h3", [t._v("Essential Links")]),
t._m(2),
e("h3", [t._v("Ecosystem")]),
t._m(3),
]
);
},
s = [
function () {
var t = this,
e = t._self._c;
return e("p", [
t._v(
" For a guide and recipes on how to configure / customize this project,"
),
e("br"),
t._v(" check out the "),
e(
"a",
{
attrs: {
href: "https://cli.vuejs.org",
target: "_blank",
rel: "noopener",
},
},
[t._v("vue-cli documentation")]
),
t._v(". "),
]);
},
function () {
var t = this,
e = t._self._c;
return e("ul", [
e("li", [
e(
"a",
{
attrs: {
href: "https://github.com/vuejs/vue-cli/tree/dev/packages/%40vue/cli-plugin-babel",
target: "_blank",
rel: "noopener",
},
},
[t._v("babel")]
),
]),
e("li", [
e(
"a",
{
attrs: {
href: "https://github.com/vuejs/vue-cli/tree/dev/packages/%40vue/cli-plugin-eslint",
target: "_blank",
rel: "noopener",
},
},
[t._v("eslint")]
),
]),
]);
},
function () {
var t = this,
e = t._self._c;
return e("ul", [
e("li", [
e(
"a",
{
attrs: {
href: "https://vuejs.org",
target: "_blank",
rel: "noopener",
},
},
[t._v("Core Docs")]
),
]),
e("li", [
e(
"a",
{
attrs: {
href: "https://forum.vuejs.org",
target: "_blank",
rel: "noopener",
},
},
[t._v("Forum")]
),
]),
e("li", [
e(
"a",
{
attrs: {
href: "https://chat.vuejs.org",
target: "_blank",
rel: "noopener",
},
},
[t._v("Community Chat")]
),
]),
e("li", [
e(
"a",
{
attrs: {
href: "https://twitter.com/vuejs",
target: "_blank",
rel: "noopener",
},
},
[t._v("Twitter")]
),
]),
e("li", [
e(
"a",
{
attrs: {
href: "https://news.vuejs.org",
target: "_blank",
rel: "noopener",
},
},
[t._v("News")]
),
]),
]);
},
function () {
var t = this,
e = t._self._c;
return e("ul", [
e("li", [
e(
"a",
{
attrs: {
href: "https://router.vuejs.org",
target: "_blank",
rel: "noopener",
},
},
[t._v("vue-router")]
),
]),
e("li", [
e(
"a",
{
attrs: {
href: "https://vuex.vuejs.org",
target: "_blank",
rel: "noopener",
},
},
[t._v("vuex")]
),
]),
e("li", [
e(
"a",
{
attrs: {
href: "https://github.com/vuejs/vue-devtools#vue-devtools",
target: "_blank",
rel: "noopener",
},
},
[t._v("vue-devtools")]
),
]),
e("li", [
e(
"a",
{
attrs: {
href: "https://vue-loader.vuejs.org",
target: "_blank",
rel: "noopener",
},
},
[t._v("vue-loader")]
),
]),
e("li", [
e(
"a",
{
attrs: {
href: "https://github.com/vuejs/awesome-vue",
target: "_blank",
rel: "noopener",
},
},
[t._v("awesome-vue")]
),
]),
]);
},
],
a = {
name: "HelloWorld",
props: {
msg: String,
},
},
u = a,
c = r(656),
h = (0, c.A)(u, i, s, !1, null, "b9167eee", null),
v = h.exports,
f = {
name: "App",
components: {
HelloWorld: v,
},
},
g = f,
b = (0, c.A)(g, l, o, !1, null, null, null),
I = b.exports;
(n.Ay.config.productionTip = !1),
new n.Ay({
render: (t) => t(I),
}).$mount("#app");
},
153: function (t) {
t.exports =
"";
},
},
e = {};
function r(n) {
var l = e[n];
if (void 0 !== l) return l.exports;
var o = (e[n] = {
exports: {},
});
return t[n](o, o.exports, r), o.exports;
}
(r.m = t),
(function () {
var t = [];
r.O = function (e, n, l, o) {
if (!n) {
var i = 1 / 0;
for (c = 0; c < t.length; c++) {
(n = t[c][0]), (l = t[c][1]), (o = t[c][2]);
for (var s = !0, a = 0; a < n.length; a++)
(!1 & o || i >= o) &&
Object.keys(r.O).every(function (t) {
return r.O[t](n[a]);
})
? n.splice(a--, 1)
: ((s = !1), o < i && (i = o));
if (s) {
t.splice(c--, 1);
var u = l();
void 0 !== u && (e = u);
}
}
return e;
}
o = o || 0;
for (var c = t.length; c > 0 && t[c - 1][2] > o; c--) t[c] = t[c - 1];
t[c] = [n, l, o];
};
})(),
(function () {
r.d = function (t, e) {
for (var n in e)
r.o(e, n) &&
!r.o(t, n) &&
Object.defineProperty(t, n, {
enumerable: !0,
get: e[n],
});
};
})(),
(function () {
r.g = (function () {
if ("object" === typeof globalThis) return globalThis;
try {
return this || new Function("return this")();
} catch (t) {
if ("object" === typeof window) return window;
}
})();
})(),
(function () {
r.o = function (t, e) {
return Object.prototype.hasOwnProperty.call(t, e);
};
})(),
(function () {
var t = {
524: 0,
};
r.O.j = function (e) {
return 0 === t[e];
};
var e = function (e, n) {
var l,
o,
i = n[0],
s = n[1],
a = n[2],
u = 0;
if (
i.some(function (e) {
return 0 !== t[e];
})
) {
for (l in s) r.o(s, l) && (r.m[l] = s[l]);
if (a) var c = a(r);
}
for (e && e(n); u < i.length; u++)
(o = i[u]), r.o(t, o) && t[o] && t[o][0](), (t[o] = 0);
return r.O(c);
},
n = (self["webpackChunkhello"] = self["webpackChunkhello"] || []);
n.forEach(e.bind(null, 0)), (n.push = e.bind(null, n.push.bind(n)));
})();
var n = r.O(void 0, [504], function () {
return r(723);
});
n = r.O(n);
})();
//# sourceMappingURL=app.6a7bde24.js.map