Duolingo English Testを受けてみた

会社で英語を使う必要が出てきたり、もっと洋書をスラスラ読みたい、英語圏で働くことも視野に入れられるようにしたいというモチベーションが出てきたので、本腰入れて英語学習をし始めました。

そこでまずは自分の実力はどれぐらいなのかを測定する必要があると思い、今回初めて「Duolingo English Test」を受けてみました。

このテストは語学学習サービスを開発しているDuolingoが開発したもので、多くの海外の大学やカレッジ、高校の入学選考書類として受け付けられるほどしっかりとしたものなので、気になっている人がいればぜひ試してもらいたいです!

englishtest.duolingo.com

何が良い?

お手軽

とにかくオンラインで手軽にいつでもテストを受けられるのが良いです!

TOEICだと申し込みの受付日程が決まっているし、実際に受験会場に足を運ばないといけないのがかなり億劫です。

でも、Duolingo English Testならオンラインでチケットを買って、いつでもテストを受けられるのですごく楽なのです!

テストの所要時間が短い

Duolingo English Testなら約60分程度で終わります。

これなら平日でも休日でもサクッと受けることができるので、休日が半日テストで潰れるなんてことがないのが嬉しいところです!

実践的な英語力を測れる

Reading, Listening, Writing, Speakingすべてが求められる問題になっています。

画像について英単語50語以上で説明、あることに対してのあなたは賛成するか反対するかの意見をスピーキングする等、個人的にこういうテストはあまり受けたことがなかったので新鮮でした!

感想

スコアは90でギリギリ中上級なんですが、見てもらうと分かるようにProduction(書いて話す能力:発信力)が著しく低いです。

今後は英語を書く・話すことを重点的に勉強しつつ、スコアを120ぐらいまで伸ばせるように毎日勉強していこうと思います。

certs.duolingo.com

blog.duolingo.com

転職して3ヶ月経った

2022年1月に今いる会社に転職して、もう3ヶ月経った。

今いる部署は技術研究をもとにプロダクトを開発することが多いので、暇なときは暇で忙しいときは忙しいらしい。

が、最初の2週間ぐらいは様子見で細かいタスクを振ってもらったが、それ以降すぐに既存アプリケーションのリアーキテクトプロジェクトに参加させてもらえたり、現在はそれと並行で新しいプロジェクトの管理画面を全部任せるね!って言ってもらえたりでやることが多いのでめちゃめちゃ楽しい!

既存アプリケーションに関しては、性能の事情から部分的にRustを使ってみようか!ということになり、CQRSアーキテクチャパターンで作られたアプリケーションのReadModelUpdater部分をRustで実装するタスクを任せてもらった。

Rustに関してはチュートリアルを軽くやったことしかなかったので、最初はとにかく書いてもコンパイルが通らない、どうやって書けばコンパイルが通るのか本当に分からずで苦労した(今もすんなりコンパイルは通せないから何回も書き直したりするけど、前よりマシだと思う)

また、非同期でメッセージの処理をしたり、Graceful Shutdownの実装のためにtokioを使ってごにょごにょやっていて、これも型の問題でいろいろ苦労した。

あと、そもそものモデリング力不足や実装力の問題で丁寧にレビューもらったりした(上長には感謝しかない🙏)

前職でコードを書いたり本読んで学んでいたつもりだったけど、環境に甘えすぎていた部分があったなと反省している。

そのあとは、Rust製のGraphQLサーバーを一部実装・修正をした。

そもそもRESTful APIしか触ってこなかったので、またここでも学び必須モードになり、ひたすらRustのライブラリであるasync-graphqlを調べたり、そもそもGraphQLってどんなもんなんだというのを調べたりしながら、実装・修正をしていた。

未だにGraphQLに関しては理解は浅いと思うし、エラーハンドリングとかのベストプラクティスってどうなのよ!っていうのはある。

新しいプロジェクトがそろそろ始まるよって段階で、上長とプロジェクト全体像のブレストしたりイベントストーミングしたりドメインイベントの発生源探したり、こういうモデルが必要だよねみたいな話をしたりして、このような取り組みはかなり意味のある時間だし、時間取って付き合ってもらってありがたいなと感謝している。

戦術的DDDで書かれたプロジェクトの機能追加、修正、保守といったことは経験あるけど、一からドメインイベント探してモデリングするみたいな経験は今まで正直なかったので、本当に学びしかないなと思ってやっていた。

ちなみに、新しいプロジェクトで書く言語はGoなのでこれがまた楽しみである!正直、チームで使う言語に強いこだわりはなくて保守・運用が楽になったり、いまチームが抱えている問題を解決できる言語であれば正直何でも良い。それよりも、新しい言語や手法を学ぶコストが〜と言われて選択肢を狭められる方が嫌だから。

とにかくいろいろ雑に書いたけど、新しい環境を楽しめていると思う。そして、チームメンバーも良い人が多いし、やっていき・のっていき精神を持つ人が多いからこういうことを取り入れよう!学ぼう!と言いやすいチームだなと思う。

今の環境は自分があまり触れてこなかった技術に触れる機会が多すぎるので毎日学ぶことが多い。というか学ばないとついていけないんじゃないか?という不安は常にあるけど、それを乗り越えてチームにより貢献できる人になりたいとは思っている。あと大事なことだけど、ちゃんと給料上げてもらいたいから学んでそれを仕事につなげて結果を出していきたいところ(お金はすべてを癒すのさ)

自分もいい歳なので、チームを技術面でもっと引っ張っていけるように技術を研鑽して日々を積み重ねていきたい所存です。

M1 Mac(arm64)環境でsbt-protocがエラーになる

M1 Mac(arm64)でプロジェクトの環境構築をしているとsbt-protocでエラーが出たので、調べた内容と解決方法を残しておこうと思います。

前提

  • MacBook Pro(arm64)
  • sbt v1.6.1
  • sbt-protoc v1.0.2

原因

protobuf3.17.3 以前のバージョンではM1 Mac向けのbinaryを配布していていないのが原因です。

sbt-protocv1.0.2のコードを見てみると、PB.protocVersion3.15.6 に指定されているため、M1 Macではエラーになっていました。

原因について言及されているIssue github.com

sbt-protocのv1.0.2v1.0.5の差分 github.com

protobufのM1 Mac Suuportについて言及されているIssue

github.com

protobuf3.17.2のmavenのリポジトリを見てみると、M1 Mac向けのバイナリが配布されていないのが分かります。

repo1.maven.org

解決方法

sbt-protocのバージョンをv1.0.5 以上に上げてください。

sbt-protocのv1.0.5PB.protocVersionを確認すると3.19.1に指定されているため、これ以上に上げればM1 Macでもエラーにならずに使用できると思います。

github.com

僕は現時点で最新のv1.0.6にアップデートすることでエラーを解消しました。(2022/1/16 時点)

『一瞬で判断する力 私が宇宙飛行士として磨いた7つのスキル』を読んだ

転職先の入社前面談の際に教えてもらった本で、その場でポチってすぐ読んでみました。

宇宙飛行士である若田光一さんが書かれた本です。

若田さんは日本人初のコマンダー(船長)を務めた方で、宇宙に旅立つ前の訓練での出来事や、実際に国際宇宙ステーション(ISS)でコマンダーとして仕事をされた体験などを交えながら、一瞬で判断するために必要なことを7つのキーワード(テーマ)を用いて書かれています。

ここがおもしろい!

1. 一般の会社や組織で起きそうな問題は宇宙飛行士の方たちにも起きるということ

宇宙飛行士の仕事っていうのは全く何をするか知らなかったので、漠然とすごいことをしているんだろうな、めちゃめちゃ能力が高い人たちだから自然と上手くいきそうだよな、とどこか別の次元の人たちのように個人的に思っていました。

ただ本を読み進めると、会社や組織で起こる問題は宇宙飛行士やその関係者の間にも起きるし、僕たちと同じようにどう解決するか頭を悩ませるということです。

例えば、コマンダーとして仕事をしていると、地上管制官と人たちとISSの宇宙飛行士の間で意見が食い違うことがやはりあるそうです。こういう問題は、普段の人間関係の間でも起こり得ることなので、その場合はどういう風に解決したのかというのは自分の体験を思い出しながら読むと大変面白かったです。

2. 不確定要素への対応

宇宙での作業は危険なことが多いので、事前にマニュアル・ドキュメントをしっかり書いて読み込むのだそうです。

また、想定外にも対応できるようにできるだけ不確定要素を減らしたり、もし複合的なトラブルが発生しても解決できるように「たぶん」や「〜ではないだろうか」で判断せず、予知できるレベルにまで体制を整えるそうです。

このような仕事のやり方はしっかり見習いたいのと同時に、エンジニアがサービスをリリースするときにどこか似ているなと感じました。

アプリケーションの設定は大丈夫か?、メモリの容量は適切か?、リクエストがこれぐらい来ると予測しているが、スパイクにも耐えられるような体制にもしておくか等、不確定要素に対しての対応を考えるというのは参考になるなと読んでいて思いました。

3. 学びについて

第2章の「学ぶ」の内容がものすごく良かったので少しだけ記載します。 宇宙飛行士の方たちは、わからないこと・曖昧なことがをそのままにすると死に直結するため、ここまで徹底的にやられているのだと思います。

新しい環境に飛び込んだときは質問しづらい気持ちになったりすると思います。また、歳を取ると分からないことに対して今更質問するのが恥ずかしいとか、まぁいいやという気持ちになってくるのかもしれません。

そんな気持ちになったときは、この本の内容を思い出して積極的に発言・質問していきたいなと思いました。

  • 愚かな質問はない
    • 「理解したつもり」が危ない
  • 質問は、自らの理解を深めるためでもある
  • 自分が感じた疑問点をそのまま残しておいてはいけない
  • 何よりも「質問をする」という行為自体が、自分の頭で考え、「理解できている部分」と「そうでない部分」を明確に分別することで生まれ出るのも事実である
  • 「分からないまま」に甘んじない
  • あやふやな記憶は、トラブルのもととなる
  • 急いで仕事をススメなければならないときに忘れてはならないのは、「何のためにやっているか」を思い出すこと
  • 長い目で見ると、速く理解することよりも重要になるのは、長い時間をかけてでも、キチンと正しく理解すること
    • 本当の意味でその知識を咀嚼していなければ意味がない
    • 中途半端な身につけ方をしていれば、結局またいつか学び直さなければならなくなる
  • 失敗を隠さず、失敗に対して常に鋭敏にアンテナを張り、失敗から何かを学ぶ姿勢が重要
  • 同じ失敗を繰り返さない
    • 一回目の失敗は、ときには寛容に受け入れてもらっても、二回目に同じ失敗をすると信頼は失墜する
    • 自分が犯した失敗を真摯に受け止め、その理由をきちんと分析し、とるべき最善の対応策を検討して習得することを怠らなければ、今後、類似した状況で同じような失敗を繰り返す可能性は少なくなる

本の内容について

最初、タイトル通り「一瞬で判断する力」について書かれていたのかと思っていましたが、ざっくり以下のような感じです。

  • 仕事への向き合う姿勢
  • 学び方
  • ものごとを前に進める方法
  • チームワーク
  • 信頼関係の構築方法
  • 決断する方法
  • 恐怖への向き合い方
  • リーダーシップ

勘違いしてほしくないのが、方法論が書かれているわけではなく、若田さんの実体験とこう行動したのが良かったというのが書かれています。

なので、ここから自分ならどうするのか、自分なりに咀嚼する必要がある本だと感じています。

どのような人におすすめか?

新しい環境に飛び込む方、マネジメントをされている方、人間関係で悩まれている方などでしょうか。

最後に

久しぶりにおもしろくて一気読みしてしまいました。

このブログでおもしろさを伝えきれないと思うので、ぜひ手にとって読んでみてください!

AWS Certified Cloud Practitionerに合格した

2021年9月26日(日)にAWS Certified Cloud Practitioner(CLF)に合格しました!

2021年9月24日時点でCLFを残し、その他は合格していたので思い切ってほぼ勉強期間を設けずに試験に臨みました。

といってもさすがに不安だったので、前日にAWSが提供するCLFの対策デジタルトレーニングやサンプル問題を解いて、当日は本屋さんでCLFの対策本を立ち読みしました。

問題自体が普段受けていたAssociateやProfessional、Specialtyに比べて具体性がなくてとっつきにくい印象を持ちましたがなんとか合格することができました!

www.credly.com

2021年の個人目標として「AWS11冠」を掲げて頑張ってきたので、とにかく年内に定めた目標を達成できてよかったなと思います。

11冠を取得して振り返りたいこともあるので、これは別のブログ内容として書こうかなと思います。

以上!

AWS Certified Advanced Networking - Specialtyに合格した

本日、2021年9月24日(金)にAWS Certified Advanced Networking - Specialty(ANS)に合格しました🎉🎉

スコアとセクションごとのパフォーマンス

今回受けたANSのスコアは 752 で合格ラインギリギリでした!

f:id:ryskit:20210924234413p:plain

そして、セクションごとのパフォーマンスは画像のとおりです。

さすがにギリギリ合格なので、改善が必要な部分が多いです。

f:id:ryskit:20210924234749p:plain

実際に試験を受けてみた感想

ANS関連のサービスのドキュメントやDirect Connectのブログ記事など読み漁り模擬試験も3周ほどしてほぼ満点まで取れるようにしたのですが、ちゃんと理解できていなかったようで本番試験の問題は上手く解けませんでした。

やはり、Direct ConnectやVIF、BGP、Site-to-Site VPNなど学んだことをうまく整理できてなかった感が否めません。

もともとインフラエンジニアではないので出てくる用語やサービスについては具体的なイメージを想像しづらく、ドキュメントに書いていることを暗記するだけになってたのが良くなかった気がします。

今後どうするか?

正直、Direct Connect等は構築することはほぼないと思うので興味が出ないのですが、それ以外のネットワークのサービスだったり知識に関しては勉強したいなと思うので引き続き継続して書籍やドキュメントを読んで理解を深めたいと思います。

最後に

AWS ANSでやっと10冠取得できました。

あとは、一番簡単なクラウドプラクティショナーを取得できれば今あるAWS資格をコンプリートできるので引き続き頑張ろうと思います!

手動でダウンロードしたsbtをsbtenvで管理する

もともとはhomebrewでsbtをインストールして使っていましたが、sbtの複数のバージョンを使って色々検証したくなりsbtenvを使ってみることにしました。

sbtenvでsbtをインストールしようとすると僕の環境ではgpg周りでsignatureの検証が失敗してしまい、すぐに解決できそうになかったので、手動でsbtをダウンロードしてsbtenvで管理する方法を取りました。

手順

前提として、xxxenv系のツールはanyenvで管理しているため、直接sbtenvをインストールして使う場合と少しパス等が違う可能性があるのでご注意ください。

1. anyenvでsbtenvをインストールする

$ anyenv install sbtenv

以下のコマンドを実行して、sbtenvのバージョンが返ってこればOK!

$  sbtenv -v

sbtenv 0.0.24

2. sbtの公式ページからsbtをダウンロードする

公式ページから sbt 1.5.5.tgz をダウンロードします。

www.scala-sbt.org

f:id:ryskit:20210923165209p:plain

3. sbtenvのversionsにディレクトリを作成する

~/.anyenv/envs/sbtenv/versions 配下にsbtのバージョン番号と同じディレクトリを作成します。

$ mkdir -p  ~/.anyenv/envs/sbtenv/versions/1.5.5

4. ダウンロードしたsbtをファイルを解凍する

tar xf ~/Downloads/sbt-1.5.5.tgz -C ~/.anyenv/envs/sbtenv/versions/1.5.5/

5. sbtenvで管理されているか確認する

以下のように、sbtenv versions を実行して、1.5.5の横に*が付いていたらOKです。

$ sbtenv versions

  1.2.8
  1.3.9
* 1.5.5 (set by /Users/[username]/.anyenv/envs/sbtenv/version)

付いていなければ、以下を実行してください。

$ sbtenv global 1.5.5

6. sbtの情報を表示してみる

welcome to sbt 1.5.5 と表示されていればOKです。

$ cd ~ && sbt about

warning: ignoring extraneous `sbt-` prefix in version `sbt-1.5.5`
         (set by /Users/[username]/project/build.properties)
[info] welcome to sbt 1.5.5 (Amazon.com Inc. Java 11.0.3)
[info] loading global plugins from /Users/[username]/.sbt/1.0/plugins
[info] loading project definition from /Users/[username]/project
[info] set current project to ryskit (in build file:/Users/[username]/)
[info] This is sbt 1.5.5
[info] The current project is ProjectRef(uri("file:/Users/[username]/"), "username") 0.1.0-SNAPSHOT
[info] The current project is built against Scala 2.12.14
[info] Available Plugins
[info]  - sbt.ScriptedPlugin
[info]  - sbt.plugins.CorePlugin
[info]  - sbt.plugins.Giter8TemplatePlugin
[info]  - sbt.plugins.IvyPlugin
[info]  - sbt.plugins.JUnitXmlReportPlugin
[info]  - sbt.plugins.JvmPlugin
[info]  - sbt.plugins.MiniDependencyTreePlugin
[info]  - sbt.plugins.SbtPlugin
[info]  - sbt.plugins.SemanticdbPlugin
[info] sbt, sbt plugins, and build definitions are using Scala 2.12.14

参照

github.com

www.scala-sbt.org

Circeでエンコード時にnullではなくフィールドごと削除する方法

Circeのエンコードでちょっとハマって時間を溶かしたので備忘録として書いておきます。

やりたいこと

あるクラスにOption型のフィールドが含まれていて、そのフィールドが存在しない(Noneである)場合はエンコード時にそのフィールドをnullではなくフィールドごと削除したい。

可能であれば余計な記述は少なくしたい。

{
  "id" : 1,
  "name" : "ryskit"
  "age": null ← 値がない場合はこのフィールドを削除したい
}

解決策

その1

以下のように オブジェクト Response 内で io.circe.Encode トレイトを実装する。

これでも実現できますが、わざわざ書くのが面倒です。

package app

import io.circe.{Encoder, Json}
import io.circe.generic.semiauto.deriveEncoder
import io.circe.syntax._

case class Response(id: Int, name: String, age: Option[Int])

object Response {
  implicit val encoder: Encoder[Response] = new Encoder[Response] {
    private val generated = deriveEncoder[Response]
    override def apply(a: Response): Json =
      generated(a).mapObject(_.filter {
        case (_, v) =>
          !v.isNull
      })
  }
}

object Main extends App {
  val response = Response(1, "ryskit", None)
  val responseJson = response.asJson.spaces2
  println(responseJson)
}

その2 (最終的な解決策)

その1と同じように、オブジェクト Response 内で encoderを定義します。

Encoderを実装せずにio.circe.generic.semiauto.deriveEncoderを使用し、dropNullValuesでフィールドを削除します。

これで記述量も少なくなります。

package app

import io.circe.Encoder
import io.circe.generic.semiauto.deriveEncoder
import io.circe.syntax._

case class Response(id: Int, name: String, age: Option[Int])

object Response {
  implicit val encoder: Encoder[Response] = deriveEncoder[Response].mapJson(_.dropNullValues)
}

object Main extends App {
  val response = Response(1, "ryskit", None)
  val responseJson = response.asJson.spaces2
  println(responseJson)
}

github.com

『インターフェイス指向設計 アジャイル手法によるオブジェクト指向設計の実践』を読んだ

どんな本なのか

インターフェイスを用いてどのようにアプリケーションを設計するか詳しく解説された本です。

インターフェイスとは何かから始まり、原則や契約、インターフェイスの評価基準、継承とインターフェイスとの対比について、ピザ屋さんを例にして説明しています。

2008年に出版された本なので古い情報について書かれている部分はありますが、現代のアプリケーション設計でも活かせる内容が書かれていると思います。

以下、目次になります。

  • 1章 インターフェイスとは何か
  • 2章 インターフェイスと契約
  • 3章 インターフェイスと材料
  • 4章 何をインターフェイスに持たせるべきか?
  • 5章 継承とインターフェイス
  • 6章 リモートインターフェイス
  • 7章 開発プロセスの開始
  • 8章 リンクチェッカー
  • 9章 Web自動集約ツール
  • 10章 サービレポジトリ
  • 11章 パターン

目的

ソフトウェア設計について学ぶことで、テストしやすく・変更しやすい、実装の意図が把握しやすい等、継続的な開発を容易に行える設計を今よりもできるようになりたいからです。

知識や経験がないと開発の際に考えられる幅が小さくなりがちなので、そこは継続的に改善していきたいところです。

感想

具象クラスではなくインターフェイスに依存させて実装してテストの際は差し替えられるようにしたりと設計・実装してきましたが、インターフェイスの原則だったり契約や評価基準というものは知らなかったので、なるほどと思いながら読んでいました。

無意識にしていたことがこの本を読んで言語化されたこともあるので、この本を読んで良かったなと思います。

読書ノート

  • ここでの契約は、インターフェイスのユーザーと実装の間に成立する約束事を指す
  • 3つの原則
    • 原則1: インターフェイス実装は、そのメソッド名が示す通りの処理をしなければならない
      • 実装はインターフェイスの作成社が意図した通りの処理を実行できなくてはいけない
      • 実装は、戻り値を適切に処理できなくてはいけない
      • 原則1は、「サブタイプの古い米はその継承元と異なってはならない」というリスコフの置換原則(Liskov Substitution Principle: LSP)に対応している
    • 原則2: インターフェイス実装は他に危害を加えてはならない
      • ここでいう危害とは、実装がプログラム内の他モジュールや、別のプログラムの正常な動作を妨害してしまうような場合のことを指す
      • 実装は、リソースをやたらに消費してしまうようなものであってはいけない
    • 原則3: インターフェイス実装は、責務を果たせない場合にそれを呼び出し元に伝えねばならない
      • 実装は、問題に遭遇し、それを自ら修正できない場合は、その問題についての呼び出し側に報告をしなければならない
        • 報告の方法(エラーシグナル)は、リターンコードでも、例外でも構わない
      • インターフェイスが報告するエラーも、インターフェイス契約の一部
  • インターフェイスを効果的に使うためには、呼び出し側と実装側がそれぞれ契約を理解していなければならない
    • 実装が呼び出し側のどのような要求に応えるべきであるか、ということについて双方の同意が必要
  • 契約の3つの側面
    • 事前条件(precondition)
      • インターフェイスのユーザーは、メソッド呼び出し時に特定の条件が満たされていることを確認しなければならない
    • 事後条件(postcondition)
      • インターフェイス内の各メソッドは、呼び出しが適切に実行された場合に真となる条件を持つ。この保証のこと。
    • クラス不変表明(class invariant)
      • すべてのオブジェクトインスタンスが満たすべき条件を示す
      • インターフェイスにおいては、クラス不変表明は通常、特定の実装のプロパティでありメソッドではない

例)

メソッド 事前条件 事後条件
set_size() なし サイズを設定
set_toppings() サイズが設定されている トッピングを設定
set_address サイズとトッピングが設定されている 住所を設定
get_time_till_delivered サイズ、トッピング、住所が設定されている なし
  • インターフェイスのプロトコルを知ることが大切
    • プロトコルとは、実行可能なメソッドコールの組み合わせ、つまりメソッドシーケンスのまとまりのこと
    • プロトコルには、インターフェイスが作成したコールバック、生成されたイベント、呼び出されたオブザーバも示される
  • インターフェイスの機能を分離させることによって、1つのインターフェイスが持つ状態遷移の数を減らすことができ、エラーの発生率を低く抑えることができる
  • インターフェイスのテストとは、対象のインターフェイ実装が契約を満たしていることを確認するために行われるべきものだと言える
  • Christine MinginsとJean-Marc Jezequelは、契約には複数のレベルがあることを提唱している
    • 型を持つプログラミング言語における、基本となる型の契約
    • 事前条件と事後条件を含む意味的な契約
    • リアルタイムシステムにおける、パフォーマンスの契約
    • 定量化の難しい、サービスの質についての契約
    • ※テストは、こうした異なるレベルの契約それぞれに対して行う必要がある
  • データインターフェイス
    • 本書では、たくさんの属性をもつクラスに対応するメソッドで構成されるインターフェイスのこと
    • データインターフェイスのメソッドは、属性に値を設定したり、属性から値を取得したりするためのメソッド
    • 状態を持つ
      • 状態には、クラスの全属性の値が含まれる
  • サービスインターフェイス
    • 本書では、渡されたパラメータに対する処理を行うメソッドを中心に構成されているモジュールのこと
    • 渡されたパラメータを処理の対象とし、実装の属性は対象にはならない
  • Ivar Jacobsonは、オブジェクトのステレオタイプを3つ定義
    • エンティティオブジェクト
      • オブジェクトの永続的な状態を示す
    • バウンダリオブジェクト
      • システムやアクター(ユーザーおよび外部システム)とのやりとりを司る
    • コントロールオブジェクト
      • 特定の仕様に対する振る舞いを表し、バウンダリオブジェクトとエンティティオブジェクトとの間をやりとりして処理を実行する
  • インターフェイス実装は、ステートフル/ステートレスにできる
    • ステートフルインターフェイス
      • メソッドは現在の状態によって異なった動作をする
      • 状態は、メソッド呼び出しの順序によって変化する
      • 長所
        • パラメータリストが短くて済む
      • 短所
        • メソッドコールの順序が重要になる
    • ステートレスインターフェイス
      • 振る舞いがメソッド呼び出しの順序によって変わることはない
      • 長所
        • メソッドコールの順序にこだわらなくてもよい
      • 短所
        • パラメータ数が多くなる
  • インターフェイスの書くメソッドは互いに高い凝集度を持たねばならない
    • 各メソッドが提供する機能は共通するコンセプトに基づかねばならないということ
  • いくつかの処理をまとめて行うようであれば、それらは同じインターフェイスで扱うべき
  • 結合度(coupling)は、あるモジュールが他のモジュールにどれだけ依存しているうかを示す
  • 実装ではなくインターフェイスに依存しているのであれば、あるメソッドが別の実装に強く結合することは稀になる
  • 疎結合とは、呼び出される側のインターフェイスの実装を変更した際に、呼び出し側のコードを変更する必要がないということ
    • 密結合になっていると、コードを変更しなければならない
  • インターフェイスの評価基準
    • 最小
      • Pros: メソッド数が少ないので、実装とテストが簡単
      • Cons: 特定の機能を実現するためにコードを書く必要があり、コードの重複を招くこともある
    • 完全
      • Pros: すべてのメソッドを利用できる
      • Cons: メソッド数が多いため、インターフェイスの理解が難しくなる
    • 単純
      • Pros: 共通する機能を簡単に使える
      • Cons: 変化が必要なときに、新しいメソッドのコーディングが必要になる
    • 複雑
      • Pros: 「好みの方法」を柔軟に選択できる
      • Cons: インターフェイスを理解することが難しい
  • インターフェイスは完全性が高まれば高まるほど、メソッドの数は増え、実装は難しくなる
  • 凝集度の高いインターフェイスを設計することが大切
    • ただし、何をもって凝集度が高いと判断するのか
  • 疎結合を目指す
  • ポリモーフィズムとは、あるメソッドの実装を複数のクラスが提供することで実現されるものを指す
    • 実現方法は2つ
      • 継承
      • インターフェイスを用いる方法
  • 複数インターフェイスの難しい点は、各実装のメソッドに共通化できるコードがでてしまうこと
    • 避ける方法は2つ
      • 方法1
        • ヘルパークラスによる処理の委譲
      • 方法2
        • インターフェイスを実装したクラスを作成
          • このクラスはメソッドの多くの実装を提供
        • インターフェイスを実装する代わりに、この作成クラスを継承する
  • インターフェイス
    • 長所
      • クラス階層を構成する経験が不十分な場合にそれを避けることができる
      • 軽装を横断できる
      • 使い方の共通性によってまとめられる
      • 改装を横断するロールを表しやすい
      • どのメソッドを実装すべきかが明白になる
      • 別の軽装改装にあるクラスがインターフェイスのサービスを提供できる
    • 短所
      • 共通機能を提供するために、ヘルパークラスがないとコードが重複する
      • ヘルパークラスを多く作成しなければならない
  • 継承
    • 長所
      • 共通の処理の委譲が少なくて済む
      • 共通の属性を取得できる
      • 実装する処理の共通性によってまとめられる
      • 基底クラスを通じて共通の実装を提供できる
      • 継承と実装が簡単
    • 短所
      • 状況の変化に柔軟に対応できない場合がある
      • ロールの変更が難しい場合はがる
  • インターフェイスは、振る舞いの共通点を表す
  • 継承は、振る舞いの共通点に加えて、実装の共通点を表す
  • ソフトウェアプロジェクトには、ビジョンが必要
  • ビジョンとはソフトウェアの使用目的を1文か2文で簡潔に言い表したもの
  • ビジョンを明確にしたら、次はソフトウェアの要件を定義する
  • 要件を表す形式には、正式の要求仕様書や、ユーザーストーリー(誰々は何々ができる)、ユースケースなど様々なものがある
  • ユースケースには、システムに対して要求を行う人と、システムから利益を受けるエンドユーザーを記述する
  • ユースケースの定義
    • ビジネスゴールを達成するために、アクターとシステムとの相互作用を定義したもの
    • アクターとは、システムの外部にいて、システムとやり取りする主体のこと
    • ユーザーを表現するクラスやユーザーが担うロール、他システムなどがアクターとなりうる存在
  • アクターの特定は、ユーザーが行う具体的な作業やユーザーの組織内での役職だどではなく、システムにおいてユーザーが担うロールに基づいて行う

2021年8月10日 ~ 16日 振り返り

振り返る。

仕事

とある機能のリリース対応

すでに開発完了されていたとある機能の暫定対応版のリリースを行いました。

プロジェクトでは複数の機能が同時並行で開発されており、開発完了したものがお客様都合でリリーススケジュールをずらしたりすることが多々あります。

今回そのうちの一つで、この暫定対応をリリースしないとAPIのクライアントであるモバイルアプリのリリースにも影響があるリリースでした。

ただ、リリース直後エラーが発生し、一旦切り戻すことになりました。

原因としては、暫定対応時に設定していた値が固定値になっており、それが原因でとある計算ロジックで落ちるというものでした。

テストパターンが少なくCIでもテストでエラーにならないために気づけず、本番環境にリリースしていくつかのパターンのリクエストが来て判明しました。

リリース直後にエラーレートが徐々に上昇したため、ログを調査するとそのような原因だったためすぐに前のアプリケーションバージョンに切り戻すことになりました。

サーバサイドチームは何人か夏休みを取得しており人が少なかったため、積極的に調査・バグの修正、再リリースの調整等を行い、無事スプリント最後の日にリリースを完了できました。

検証環境もありCI/CD環境もあるにも関わらず、このバグを見逃したのはテストデータの網羅性が足りなかったのと適切なクラスにテストケースを追加できていなかったからです。

また、クラス設計等に問題がある可能性もあります。とある修正をしたときに正しく計算ロジックがテストで実行されていればバグに気づけた可能性があります。

今後、システムの安定稼働、開発・運用の容易性を高めるためにリファクタリングは行っていきたいと思います。

Lambdaにリソースタグを付与する

プロジェクトでは開発したリソースにタグを付与して、CloudWatchやDatadogで監視しやすいようにしています。

Serverless Frameworkでリソース作成、デプロイをしているLambdaにはこのリソースタグが付与されていなかったため付与する対応を行いました。

以下のようにStackTagsを設定すればLambdaに付与されます。

stackTags:
  key: value
iam:
  # Overwrite the default IAM role which is used for all functions
  role: arn:aws:iam::XXXXXX:role/role

www.serverless.com

プライベート

読書

  • ドメイン駆動設計入門 - ボトムアップでわかる!ドメイン駆動設計の基本

blog.ryskit.com

その他

UdemyでAkka Persistenceの動画を見終わりました。

内容も分かりやすく面白いのでおすすめです!

www.udemy.com