1
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

panda(Kotlin, Test, BlockChain)Advent Calendar 2022

Day 14

Deno入門[インストール、環境構築、ファイル実行、標準ライブラリ]

Last updated at Posted at 2022-12-13

この記事は筆者のソロ Advent Calendar 2022 14日目の記事です。

前回までKotlinとテスト中心に記事を書いてきましたが今回からJavascriptランタイムであるDenoに入門してみたのでその備忘録です。

Deno入門[インストール、環境構築、ファイル実行、標準ライブラリ] <- 今ここ
Deno入門[CLIコマンド、モジュール、環境変数、Webフレームワーク]
Deno入門[npmモジュールの使用]
Deno入門[prisma + oakで作るAPI]
Deno入門[テスト編]

Denoとは

Denoの情報はまだ少ないにしろ大変わかりやすくまとめていただいている神記事があるので詳しくはそちらを

公式はこちら
https://deno.land/

なぜ書くのか

普段はサーバーサイドエンジニアとしてKotlinを書いていますが、世の中のトレンド的なものを考えるとフロントエンドを書かないにしてもTypeScriptのキャッチアップをしておかないといろんなものから置いていかれそうという焦燥感と恐竜が可愛いという理由と新しいものは触っておきたいというミーハー心からDenoを使ってみようと思っていたのですがなかなか時間が取れず、Advent Calenderを利用して2022年滑り込み入門することにしました。

基本的に公式ドキュメントのサンプルなどを実際に動かしていきたいと思います。

Getting Started

install

とりあえずインストール

brew install deno

deno --version
deno 1.28.1 (release, aarch64-apple-darwin)
v8 10.9.194.1
typescript 4.8.3

好きなエディタで環境整えてねと公式に書いてあって親切。VSCode, vim, JetBrainsなどがありましたが、筆者はVSCodeの環境を整えます。

公式から拡張プラグインが提供されているのでインストール
https://marketplace.visualstudio.com/items?itemName=denoland.vscode-deno

デフォルトではdenoのプラグインは有効になっていないのでワークスペース単位で有効設定にすることを推奨しているようです。

VSCodeでコマンドパレットを開き、Deno: Initialize Workspace Configurationを選択し、設定を有効にするとプロジェクトのルートディレクトリに.vscode/setting.jsonが作成される。

{
  "deno.enable": true,
  "deno.unstable": true
}

Hello World

以下のファイルを作成しHello Worldしてみる。

first_steps.ts
console.log("Welcome to Deno!");

実行

% deno run first_steps.ts 
Welcome to Deno!

DenoはURLからスクリプトを実行することもできるらしい。

% deno run https://deno.land/std@0.103.0/examples/welcome.ts
Welcome to Deno!

fetch APIでHTTPリクエスト

標準APIであるfetch APIを使用しHTTPリクエストをしてみます。first_steps.tsに以下を追加します。

first_steps.ts
const url = Deno.args[0];
const res = await fetch(url);

const body = new Uint8Array(await res.arrayBuffer());
await Deno.stdout.write(body);

実行時引数でURLを指定し、fetchし取得したレスポンスを標準出力に書き出すだけのコードです。これを実行すると以下のようになります。

% deno run first_steps.ts https://example.com 
Welcome to Deno!
✅ Granted net access to "example.com".
<!doctype html>
<html>
<head>
    <title>Example Domain</title>

    <meta charset="utf-8" />
    <meta http-equiv="Content-type" content="text/html; charset=utf-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1" />
    <style type="text/css">
    body {
        background-color: #f0f0f2;
        margin: 0;
        padding: 0;
        font-family: -apple-system, system-ui, BlinkMacSystemFont, "Segoe UI", "Open Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
        
    }
    div {
        width: 600px;
        margin: 5em auto;
        padding: 2em;
        background-color: #fdfdff;
        border-radius: 0.5em;
        box-shadow: 2px 3px 7px 2px rgba(0,0,0,0.02);
    }
    a:link, a:visited {
        color: #38488f;
        text-decoration: none;
    }
    @media (max-width: 700px) {
        div {
            margin: 0 auto;
            width: auto;
        }
    }
    </style>    
</head>

<body>
<div>
    <h1>Example Domain</h1>
    <p>This domain is for use in illustrative examples in documents. You may use this
    domain in literature without prior coordination or asking for permission.</p>
    <p><a href="https://www.iana.org/domains/example">More information...</a></p>
</div>
</body>
</html>

上記のように実行するとリクエストは成功するのですが以下のようなエラー表示が出ます。これはDenoは標準でセキュアな作りとなっておりネットワークアクセスに対してpermissionを許可する必要があるからです。

⚠️  ┌ Deno requests net access to "example.com".
   ├ Requested by `fetch()` API
   ├ Run again with --allow-net to bypass this prompt.
   └ Allow? [y/n] (y = yes, allow; n = no, deny) > 

ネットワークアクセスを許可して実行するには以下のように実行します。

deno run --allow-net=example.com first_steps.ts https://example.com

ファイル読み込み

DenoはWebからでないAPIも提供している。例えば、ファイルシステムのAPIはWeb標準の形がないため、Denoは独自のAPIを提供しています。以下のコードはファイルを読みこみ、その内容を標準出力にコピーして出力するコードです。

reading_file.ts
import { copy } from "https://deno.land/std@0.166.0/streams/conversion.ts";
const filenames = Deno.args;
for (const filename of filenames) {
  const file = await Deno.open(filename);
  await copy(file, Deno.stdout);
  file.close();
}

実行にはAPIリクエスト同様ファイル読み込みの権限が必要です。

% deno run --allow-read reading_file.ts hello.txt hi.txt
Hello!!
Deno!!
Hi!!
Deno!!%

HTTPサーバーの構築

以下のように標準ライブラリからserver関数をimportしサーバーを立ち上げることができます。

http_server.ts
import { serve } from "https://deno.land/std@0.157.0/http/server.ts";

const port = 8080;

const handler = async (request: Request): Promise<Response> => {
  const resp = await fetch("https://api.github.com/users/denoland", {
    // The init object here has an headers object containing a
    // header that indicates what type of response we accept.
    // We're not specifying the method field since by default
    // fetch makes a GET request.
    headers: {
      accept: "application/json",
    },
  });
  return new Response(resp.body, {
    status: resp.status,
    headers: {
      "content-type": "application/json",
    },
  });
};

console.log("Listening on http://localhost:8000");
serve(handler);

以下のようにサーバーを起動

% deno run --allow-net --allow-write http_server.ts
Listening on http://localhost:8000

アクセスしてみる

% curl http://localhost:8000 | jq
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
100  1222    0  1222    0     0   4484      0 --:--:-- --:--:-- --:--:--  4628
{
  "login": "denoland",
  "id": 42048915,
  "node_id": "MDEyOk9yZ2FuaXphdGlvbjQyMDQ4OTE1",
  "avatar_url": "https://avatars.githubusercontent.com/u/42048915?v=4",
  "gravatar_id": "",
  "url": "https://api.github.com/users/denoland",
  "html_url": "https://github.com/denoland",
  "followers_url": "https://api.github.com/users/denoland/followers",
  "following_url": "https://api.github.com/users/denoland/following{/other_user}",
  "gists_url": "https://api.github.com/users/denoland/gists{/gist_id}",
  "starred_url": "https://api.github.com/users/denoland/starred{/owner}{/repo}",
  "subscriptions_url": "https://api.github.com/users/denoland/subscriptions",
  "organizations_url": "https://api.github.com/users/denoland/orgs",
  "repos_url": "https://api.github.com/users/denoland/repos",
  "events_url": "https://api.github.com/users/denoland/events{/privacy}",
  "received_events_url": "https://api.github.com/users/denoland/received_events",
  "type": "Organization",
  "site_admin": false,
  "name": "Deno",
  "company": null,
  "blog": "https://deno.land",
  "location": null,
  "email": "support@deno.com",
  "hireable": null,
  "bio": null,
  "twitter_username": "deno_land",
  "public_repos": 80,
  "public_gists": 0,
  "followers": 1683,
  "following": 0,
  "created_at": "2018-08-02T22:47:41Z",
  "updated_at": "2022-10-05T16:30:46Z"
}

まとめ

  • Denoのインストールについて紹介しました
  • DenoのVSCodeの開発環境の準備について紹介しました
  • Denoの標準ライブラリやfetchAPIの使用について紹介しました

DenoをインストールするだけでTypeScriptがすぐ書けるのがめちゃくちゃいいなと思いました。とりあえず今回は以上です!

1
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?