はじめに
「コロナ禍」と言われる言葉が生まれてから、およそ一年ほどは経ちましたね。ここ一年間ネガティブなニュースを多く耳にしたのは僕だけでしょうか?「〇〇で感染者数が過去最多」とか「コロナ禍で深刻化する〇〇」とか、不安な気持ちになるようなニュースが毎日のようにあったような気がします。実際はどうなんでしょうね。
今回は、「実際の世の中のニュースはマイナスで溢れているのか?」そんな疑問から直近1週間の景況を可視化してみようと思います。
前提
Node.js(Express)で実装
事前調査
使用API
- News API
- Natural Language API
News API
- Top headlines:最新のヘッドライン
- Everything:全ての記事
- Source :Top headlines が利用できるニュースパブリッシャーのサブセットを返す
News API には上の三つのエンドポイントが用意されています。このうち、Everything
がパラメータとして、期間を設定することが出来ます。Everything
は、感情分析の結果の推移を観測したい、という今回の要望にマッチしているため、こちらを使用したいと思います。パラメータにはこの他にも、検索元のドメインを指定したり、ソートをして人気の記事を上位に持ってきたりすることが出来るので求めたいニュースに合わせて値を設定すると良いです。
※Everything
のエンドポイントは日本語のニュースを検索対象にすることができません
Natural Language API
詳しくは公式をご覧ください。
https://cloud.google.com/natural-language/docs/quickstart-client-libraries?hl=ja
Natural Language API で分析できる感情は、score
と magnitude
の数値によって表されます。score はポジティブとネガティブを -1.0 ~ 1.0 で 表し、magnitude は感情の大きさを 0 以上で表します。これらの値は、最終的に可視化する際の指標として使用します。
開発
newsを取得する
まずは、ニュースの取得する部分の実装を行っていきます。以下の関数は日付と検索ワードを引数に指定して、人気のあるニュースを取得することができ、戻り値は後の処理のawait
で使用するためにPromiseを返すようにしてあります。
News API にはNode.js のクライアントライブラリもあるので、今回はこちらを活用して実装しました。
constNewsAPI=require('newsapi');constnewsapi=newNewsAPI('API_KEY');// ニュースの取得functionfetchNews(Date,keyword){returnnewPromise(resolve=>{newsapi.v2.everything({// optionsq:keyword,from:Date,to:Date,language:'en',sortBy:'popularity',pageSize:50,page:2}).then(news=>{resolve(news)});})}
感情分析をする
取得できたニュースの概要から感情を分析出来るように、テキストを引数とした関数を作成します。この関数は、ニュースのタイトルやURLの他に、感情分析の結果のスコアも含めた連想配列をレスポンスとして返します。こちらも、News API と同様にNode.js のクライアントライブラリを使用しています。
https://cloud.google.com/natural-language/docs/analyzing-sentiment?hl=ja
constlanguage=require("@google-cloud/language"),client=newlanguage.LanguageServiceClient();// 引数に与えられたテキストの感情を分析するasyncfunctionsentiment(text,title,url){constdocument={language:'ja',content:text,type:'PLAIN_TEXT',};varobj={"Title":title,"Url":url,"Sentence":"","Score":0,"Magnitude":0}const[result]=awaitclient.analyzeSentiment({document});constsentences=result.sentences;sentences.forEach(sentence=>{obj["sentence"]=sentence.text.content;if(sentence.sentiment.score!=0){obj["score"]=sentence.sentiment.score;}if(sentence.sentiment.magnitude!=0){obj["magnitude"]=sentence.sentiment.magnitude;}});returnobj;};
次に、二つのAPIを使用した関数を掛け合わせます。
// 取得したニュースに感情分析をかけるasyncfunctionanalysis(Date,keyword){varanalyzedList=[];varnews=awaitfetchNews(Date,keyword)for(varitemofnews["articles"]){awaitsleep()varsentenceObj=awaitsentiment(item.description,item.title,item.url);analyzedList.push(sentenceObj);}returnanalyzedList;}
この関数は、日付と検索ワードを引数とし、引数の値を使ってニュースを取得します。
取得したニュースのdescription
を感情分析APIを使用した関数の引数として与え、一件一件評価し、帰ってきた値を配列にまとめておきます。
ここでのanalyzedList
という配列には、指定した日付で取得できたニュースの情報が含まれており、一件のニュースには「タイトル」「URL」「ニュースの概要」「感情分析の結果」がまとまっています。
あとは、実際に取得する日付や期間を決めたり、analyzedList
のスコアの合計の算出やソートをするなどして、実際にダッシュボードに可視化するための情報を選択していけば完成です。
今回は、ダッシュボード内で以下のことが出来るように、追加で処理を加えました。
- 算出された1日のニュースのトータルスコアの推移を見ることができる
- 今日の日付から1週間前までの推移を観測出来る
- 解析期間中の、最も高いスコア(ポジティブ)の三件と最も低いスコア(ネガティブ)の三件のニュースをを知ることが出来る
最終的にレンダリングに使うコンテキストは以下のようになりました。
date:Array,// 解析した日付:Chart用totalScoreList:Array,// 解析した日付ごとのトータルスコア:Chart用picUp:Object,// スコアの上位三件&下位三件logs:Array,// 検索ワードと検索日の履歴searchKeyword:String// 検索ワード:ダッシュボードの見出し用
成果物
インターフェース部分は、BootStrapのサンプルに、作りたいもののイメージに近しいものがあったのでこちらを少しだけカスタマイズして使用しています。
検索窓にワードを入力して解析開始です。1週間のニュースを取得し、一件一件評価しているので、可視化までに時間がかかってしまうことが少しネックです。
直近の推移だけ見ると、ほとんど0を下回っているので、あまり良い印象では無いというが分かります。
Good Weekly Newsには「全国予防接種の日」についての記事がありますね。これは、確かにプラス面のニュースです。Bad Weekly News はバイデン政権の記事がトップに来ていました。政治の話はあまり詳しくは無いですが、執筆者の言い回し自体はかなりマイナス表現が多かったと感じるので、評価自体は妥当な値だと思います。
他のワードでも試してみます。
個人的な興味でアベンジャーズを調べてみます。(アメコミが好きなので)
流石にCOVID19よりも印象は良いみたいです。
やはり今後のCOVID19に関しては、まだまだマイナスニュースを目にする機会が多くなりそうですね。
まとめ
今回は、「世の中悪い出来事ばかりじゃ無いはず!」という思いから、現状を確認するために、News API とNatural Language API で景況を可視化してみました。
検索ワードを元にした評価を可視化しましたが、ワードによって結果が大分異なるので、シンプルにヘッドラインだけでも良かったような気がします。ただ、News API のTop headlines
エンドポイントは、期間を指定できないので、また違う仕組みでの実装が必要になってきそうです。
また、TwitterなどのSNSの情報ソースを使うのも面白しいかと思います。今はSNSもみる人の感情を左右する立派な情報源なので、よりリアルの声を反映出来るのかもしれないです。