2
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 5 years have passed since last update.

はじめてのPHPアプリ作成〜Spotify Web APIとElement UIを使って、Herokuデプロイまで〜

Last updated at Posted at 2020-04-11

これまでPHPで記述したアプリケーションをWEB上に公開したことがなかったので、この度ポートフォリオ提出用に公開しました。

今回作成したポートフォリオの詳細を述べることも兼ねて、Qiitaの記事として投稿します。

私のエンジニアとしての経歴は、「半年間ほど渋谷の自社開発企業でフロントチーム(使用言語はHTML, SCSS, JavaScript, jQuery)のメンバーとしてインターンシップ歴あり」程度のもので、普段は経済学部生として社会保障分野の研究をしています。

アプリケーションの概説

Spotify Web APIを利用して、適当なアーティストを入力するとその関連アーティストが表示されます。
加えて、ワンクリックでSpotifyへと遷移することができ、すぐに該当アーティストの楽曲を聞くことが可能です。

spotify_app1.png spotify_app2.png spotify_app3.png spotify_app4.png

ユーザーエージェントによる切り分け

レスポンシブデザインは採用せず、PCかSPかを識別して読み込むページを変更する方法を採っています。

index.php
<?php
	$ua = $_SERVER['HTTP_USER_AGENT'];
	$browser = ((strpos($ua, 'iPhone') !== false) || (strpos($ua, 'iPod') !== false) || (strpos($ua, 'Android') !== false));
    if ($browser == true) {
		$browser = 'sp';
	}
	if ($browser == 'sp') {
        //SPの場合に読み込むファイル
        require('index_sp.php');
    } else {
        //PCの場合に読み込むファイル
        require('index_pc.php');
    }
?>

フロントエンドについて

コンポーネントのほとんどはElement UIを活用しました。
Element UIを用いるためにVue.jsの環境構築をする必要がありましたが、お手軽CDNで済ませました。

VueのCDN版について

Element UIを使った理由は、インターン先で少しだけ使ったときにおもしろいなと感じていたこと、便利ツールのできることとできないことを知りたかったということが挙げられます。
ふわっとした動機です。

ちなみに、Element UIを使ってみて感じたことは、以下のとおりでした。
・ 単純におもしろい
・ リッチなコンポーネントがすぐに使えて便利
・ 細かいレイアウトを変更したいときに不便(el-hogeはひとつのタグに見えても、開発ツールでコンパイル後は複数のタグかつ複雑な構造になっていたりするため)
・ 大規模な開発には向かなそう

バックエンドについて

バックエンドについては以下のページを大いに参考にさせていただきました。

PHPでSpotify APIを使う

Spotify APIを使ってLINEBotであなたにおすすめプレイリストを全自動で作ってもらう

本稿では、以下の機能を実装するための関数について紹介します。

  1. 入力されたアーティストの情報
  2. 入力アーティストのアルバムの情報
  3. 関連アーティストのトップトラックの情報
  4. 関連アーティストの情報

また、コード内のTODOコメントは今後改善したいポイントであります。

0.下準備

Spotify for Developerにアクセスしてあれこれすること、Spotify Web APIをPHPで利用するためのライブラリをダウンロードすることについては、上述の参考サイトに丸投げします。

上記ライブラリを利用するために、以下の関数を切り分け。
(今回はユーザー情報を用いないので、REDIRECT_URIの記述は省略しました。)

spotify.php
<?php
    require_once('vendor/autoload.php');
    $CLIENT_ID = getenv('CLIENT_ID');
    $CLIENT_SECRET = getenv('CLIENT_SECRET');
    $session = new SpotifyWebAPI\Session(
        "{$CLIENT_ID}",
        "{$CLIENT_SECRET}"
    );
    $api = new SpotifyWebAPI\SpotifyWebAPI();
    $session->requestCredentialsToken();
    $accessToken = $session->getAccessToken();
    $api->setAccessToken($accessToken);
?>

CLIENT_ID, CLIENT_SECRETについては、Spotify for Developerのダッシュボードから確認してください。
また、CLIENT_ID, CLIENT_SECRETについては、環境変数ENVを参照させる形をとっています。
この際、以下のサイトを参考にさせていただきました。

HerokuでWebアプリ開発を始めるなら知っておきたいこと(5) 環境変数ENV

HerokuでPHPを使うときに気を付けるところ

$ heroku config:add CLIENT_ID='hogehoge' --app 'hugahuga'
$ heroku config:add CLIENT_SECRET='foo' --app 'bar'

// こちらで設定した環境変数については、以下のコマンドで閲覧可能です。
$ heroku config

CLIENT_ID='hogehoge'のように、左辺と右辺との間にスペースを空けずに入力してください。

CLIENT_ID, CLIENT_SECRETを取得したい場合は、getenv()で取得可能です。


以下、私が作成した関数についての紹介となりますが、関数についてはSpotifyWebAPI.mdを見ながら作成しました。
豊富なメソッドがあるので眺めているだけでもおもしろいです。

1.入力されたアーティストの情報を持ってくる関数

引数に渡されたアーティスト名を利用し、ライブラリで用意されているメソッドを通じ、そのアーティストの情報を入手します。
これにより、入力されたアーティストの情報を動的に表示させることができるようになりました。

function.php
function artistSearch($artistName) {
    global $api;
    $artistInfo = $api->search($artistName, 'artist', array('limit' => 1));
    if (isset($artistInfo->artists->items)) {
        foreach ($artistInfo->artists->items as $data) {
            $artistData = array(
                'id' => $data->id,
                'artist_name' => $data->name,
                // TODO: イメージが複数あるならランダムで表示されるようにしてもいいかも
                'image' => $data->images[0]->url,
                'artist_url' => $data->artist[0]->external_urls,
            );
        }
        return $artistData;
    } else {
        // $artistInfo->artists->items がヒットしないとき
        return false;
    }
}

global $api;としているのは、本関数の中でspotify.phpにて記述した$apiという変数を用いるためです。

2.入力アーティストのアルバムの情報をもってくる

function.php
function relatedArtistTopAlbum($artistId) {
    global $api;
    $relatedArtistAlbum = $api->getArtistAlbums($artistId, array('country' => 'JP'))->items;
     // 取得するアルバムは人気があるものとは限らない
     $relatedArtistTopAlbum = array(
         'artist_url' => $relatedArtistAlbum[0]->artists[0]->external_urls->spotify,
         'artist_name' => $relatedArtistAlbum[0]->artists[0]->external_urls->name,
         'album_image' => $relatedArtistAlbum[0]->images[0]->url,
         'album_name' => $relatedArtistAlbum[0]->name
     );
     return $relatedArtistTopAlbum;
}

3.関連アーティストの情報を持ってくる

1.入力されたアーティストの情報を持ってくる関数で得たアーティストのidを引数に、そのアーティストに関連するアーティストの情報を入手します。

function.php
function relatedArtistSearch($artistId) {
    global $api;
    $relatedArtist = $api->getArtistRelatedArtists($artistId)->artists;
    $relatedArtistSelect = array();
    // 関連アーティストをいくつ持ってくるかを定義
    $countNum = 6;
    if (count($relatedArtist) >= $countNum) {
        $selectionNum = $countNum;
    } else {
        $selectionNum = count($relatedArtist);
    }
    for ($i = 0; $i <= $selectionNum - 1; $i++) {
        $relatedArtistData = array(
            'id' => $relatedArtist[$i]->id,
            'name' => $relatedArtist[$i]->name,
            'images' => $relatedArtist[$i]->images[0]->url
        );
        // 関連アーティストを$countNumだけ取得
        array_push($relatedArtistSelect, $relatedArtistData);
     }
     return $relatedArtistSelect;
}

4.関連アーティストのトップトラックの情報を持ってくる

3.関連アーティストの情報を持ってくるで得た関連アーティストのidを引数に、そのアーティストのトップトラック(人気曲)の情報を入手します。

関連アーティストが$countNum(今回は$countNum=6)つ収録された変数$relatedArtistSelectは配列ですので、foreachで関連アーティスト各々についての情報を入手します。

ここで用いているgetArtistTopTracks()というメソッドは大変面白く、本アプリケーションの核といってもよい重要なものです。

function.php
function relatedArtistTopTracks($relatedArtistSelect) {
    global $api;
    $topTracksSelect = array();
    foreach ($relatedArtistSelect as $data) {
        $topTracks = $api->getArtistTopTracks($data['id'], array('country' => 'JP'))->tracks;
        $topTracksData = array(
            'track_id' => $topTracks[0]->id,
            'track_url' => $topTracks[0]->external_urls->spotify,
            'artist_name' => $topTracks[0]->artists[0]->name,
            'artist_url' => $topTracks[0]->artists[0]->external_urls->spotify,
            'album_name' => $topTracks[0]->album->name,
            'album_image' => $topTracks[0]->album->images[0]->url,
            'album_url' => $topTracks[0]->album->external_urls->spotify,
        );
        if (isset($topTracks)) {
            // 関連アーティスト各々に対してトップトラックを取得
            array_push($topTracksSelect, $topTracksData);
        }
    }
    return $topTracksSelect;
}

1.〜4.の関数をどう使ったのか

以上の4つの関数を以下のように用いました。

index_pc(_sp).php
<?php
// TODO: 公開前に0にする
ini_set('display_errors', 0);

// spotify web api 使用
require('spotify.php');

// 切り出した関数読み込み
require('function.php');

// アーティスト情報取得
// TODO: 空欄はエラーなので空欄のときはとりあえずなにかいれておく
if ($_POST['artistName'] === "")  {
    $_POST['artistName'] = "アルベム";
}

// 入力されたものをサニタイズ
$artistName = htmlspecialchars($_POST['artistName'], ENT_QUOTES, "UTF-8");

//入力されたアーティストの情報を取得
$artistData = artistSearch($artistName);

// 入力されたアーティストのアルバム取得
$relatedArtistAlbum = relatedArtistTopAlbum($artistId);

// 関連アーティスト取得
$artistId = $artistData['id'];
$relatedArtistSelect = relatedArtistSearch($artistId);

// 関連アーティストのトップトラック取得
$topTracksSelect = relatedArtistTopTracks($relatedArtistSelect);
?>

少々話が逸れてしまいますが、サニタイズについての理解が疎いため問題点などございましたらご指摘いただきたいです。(もちろんその他なんでも結構でございます)

Herokuにデプロイ

さて、望んでいた挙動をしてくれるようになったので、外部に公開しようということで、今回はHerokuにデプロイして見たいと思います。
本当はAWSを使ってあれやこれやしたかったのですが、またの機会に。

デプロイ方法はこちらのQiitaの記事を参考にさせていただきました。

下手にDocker-Composeを使ってphpの環境構築をしていたため、その作業ディレクトリからHeroku用のディレクトリにまるまるファイルやフォルダをコピーしまして、上記サイトの手順に従ってデプロイを行いました。

以前挫折したHerokuでしたが、上記の記事のおかげもあって、特につまずきもなくスムーズにデプロイが完了しました。

追記:
herokuとgithubが連携できるとは知らず、面倒な方法をとってしまいました。
HerokuとGitHubを連携させて自動デプロイ環境を作ろう!を参考にすれば、もっとカンタンでした。

今後やりたいこと

ひとまず最低限のものをデプロイ出来てホッとしています。
今後はAWS, Docker, CI/CDなどの技術を盛り込んでみたいと考えています。

また、フロントのポジションでインターンシップに参加していたこともあり、phpを書くのは久々でしたので良い復習になりました。

以上となります。ありがとうございました。

2
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
2
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?