21
8

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.

クソアプリ2Advent Calendar 2019

Day 12

Twitterでコードをつぶやくと実行してくれるアプリを作った

Last updated at Posted at 2019-12-11

#はじめに

本記事はクソアプリ2 Advent Calendar 12日目の記事です。
皆さんのクオリティの高いクソアプリを見て「来る場所間違えたかも…」と思っている所存です。
卒論の現実逃避としてササっと作ったものなので不備はありそうですが、温かい目で見てもらえると嬉しいです。

#概要
Twitterを見ていてふとコードが思いつくこと、ありませんか?
そんなときに、わざわざターミナルやテキストエディタを開いて…ってなると少し面倒かと思います。まじてや外出中に「この挙動を調べたいな」と思っても、街中でPCを開くことはおろかPC持ってないよ…となってしまうこともしばしば。

今回はそんな人のために「Twitterでコードをつぶやくと実行結果が返ってくるアプリ(?)」を作りました。

#完成形

TwitCodeRunnerと名付けました。
ezgif.com-video-to-gif.gif

実行結果を自分のアカウントから呟かせたい場合はTwitterDeveloperの登録が必要となります。あと、Rubyが必要です。
詳しい仕様はREADMEを参照してください。
Github

タイムラインの取得開始はruby tcr-main.rbでできます。TwitterのUserStreaming機能を用いているので、実行してからはプログラムを閉じない限りはリアルタイムで使えます。

また、たまに僕のTwitterアカウントでも運用しています。TwitterDeveloperに登録してない人は僕や他の人がプログラムを動かしているときに便乗しましょう。正直使い勝手が悪いですね…。(時間がなくてみんなが使えるようなシステムを組めませんでした、ごめんなさい!)

#おおまかな仕様
使用言語はRubyです。ツイートの読み込みや実行結果のつぶやくのはおなじみTwitterAPIを使用してます。肝心のコードの実行はPaiza.IOのAPIを用いました。これは、ブラウザ上でコードを書いてすぐ実行できるサービスで、プログラミング初級者や環境構築前にサクッとプログラミングしたいと言う方なんかは触ったことがあるのではないでしょうか。

Paiza.IOは無料でAPIを公開しています。APIを試せるページがあり、とても使いやすかったです。

対応言語もPaiza.IOに準じています。

  • Bash
  • C
  • C#
  • C++
  • Clojure
  • COBOL
  • CoffeeScript
  • D
  • Elixir
  • Erlang
  • F#
  • Go
  • Haskell
  • Java
  • JavaScript
  • Kotlin
  • MySQL
  • Perl
  • PHP
  • Python2
  • Python3
  • R
  • Ruby
  • Rust
  • Scala
  • Scheme
  • Swift
  • VB

Objective-Cだけは、Paiza.IO自体は対応してますが自分のアプリでは対応していません。

#何がクソなのか
多分「Twitterからプログラミングができる」という響きはめちゃくちゃ有用そうに聞こえます。(有用かどうかは個人の判断に任せます)開発を始める前も「天才か?」と自画自賛していました。ただ、いざできたものを見返すといくつか思うところがあります。

##有用性
個人の判断に任せると言いましたが、よくよく考えると本当に使い道なくね?と思ってしまいます。完成形のGIFはPCからTwitterを動かしているのですが、一応スマートフォンからもできることはテスト済みですので、Twitterから離れることなくプログラミングができます。ここがアピールポイントです。

##テストが間に合ってない
これはアプリ自体の仕様というよりは僕自身の問題なのですが、単純に28言語分のテストをしている時間や費やす労力がないです。なのでこれを利用しようとしてもどこかでエラーを起こすかもしれないです。Githubにコードを置いておくので、拡張したり利便性を求めてさらに何かしてみようという方はご自由にどうぞ。自己責任で!(そのまま使う分には許諾などいりませんが、機能拡張や多くの人に向けてのサービス化などがもしもありましたら一言くれると『使ってくれる人いるんだ…!』と思って嬉しくなります。)

##そもそも二番煎じ
paiza_run というTwitterからコードを実行できるサービスがありました。名前の通り、公式です。ただ、2019年11月現在このアカウントは凍結中でした。どうやらpaiza_runはQuineでの無限再帰にハメられた挙句おもちゃにされた過去があるようです。

##実行が遅い
外部APIを用いているためです。

#ここがすごい
では、paiza_runにはない新規性を出すとしたらどこか。

Twitterの1つの投稿に対しての文字数制限は140字で、これ以上長いコードは実行できなかったようです。(僕がpaiza_runを知ったのはこのアプリを開発し始めた後なので正確な機能はわかりません><)

しかし、TwitCodeRunnerは、自分に対してリプライする機能を用いて文字数制限が無くなりました。これについては後述します。
ezgif.com-video-to-gif-2.gif

#詳細な仕様

ここからは興味があるorマサカリを構えた人向けです。Rubyは全て独学で勉強しているので、しっかりできる人からしたら変なことをしているコードがあるかもしれません。「こうした方がいいよ!」などアドバイスを頂けたらと嬉しいです。(ただ、いじめるのはやめてください

##実行の流れ

まず、TwitCodeRunnerは、twitcoderunner (適当な文字列).(拡張子)というツイートがトリガーとなって作動します。これが実行コマンドだと思ってください。拡張子の部分は、実行したい言語のものを入れてください(例えばRubyならrb、Clojureならcljなど)

twitcoderunner (適当な文字列).(拡張子)と入っているツイートを正規表現によって見つけ出すと、そのツイートのIDとコードを書いた人のユーザーネームを取得します。取得はTwitterAPIのstatuses/filterを用いてUserStreamでリアルタイムで頑張ってます。statuses/filterの仕様上、取得できるのは公開アカウントのツイートのみになります。基本一人で使用することを前提としての処理なのでこうしてますが、コードを書いた人と実行コマンドを呟いた人が違う場合はコードを書いた人に実行結果が届くので注意してください。しかも僕から直々にです。

実行コマンドの入ったツイートのIDを受け取ったあと、ソースコードを取得・再構築してrunnerという関数で動かします。

main.rb

require_relative '../Privatekey/oauth_twitter'
require_relative 'runner'
require_relative 'apifunc'
require_relative 'sourcejoin'
require 'json'
require 'net/http'

consumer = OAuth::Consumer.new(
  @client.consumer_key,
  @client.consumer_secret,
)
endpoint = OAuth::AccessToken.new(consumer, @client.access_token, @client.access_token_secret)

filter_option = {
  track: "twitcoderunner"
 }

reg1 = /\Atwitcoderunner (.*)\.(.*)\Z/

@client_s.filter(filter_option) do |object|
  if reg1 =~ object.text
    lang = uselang($2)
    responce = endpoint.get("https://api.twitter.com/1.1/statuses/show/#{object.id.to_s}.json")
    result = JSON.parse(responce.body)
    finid = result["id_str"]
    name = result["in_reply_to_screen_name"]
    sourcetext = sourcejoinfirst(finid)
    source = charrefconversion(sourcetext)
    runner(source, lang, name)
  end
  puts "---------------------------------------------"
end
runner.rb
require_relative '../Privatekey/oauth_twitter'
require_relative 'apifunc'
require 'json'
require 'net/http'

def runner(source,language,screen_name)
  if source == "error[NO TEXT]"
    puts "実行できるファイルがありません"
  elsif language == "none"
    puts "拡張子に対応した言語がありません"
  else
    uri = URI.parse('http://api.paiza.io/runners/create')
    param = {
      language: language,
      source_code: source,
      api_key: :guest
    }
    #------------------create code------------------
    responce = Net::HTTP.post_form(uri, param)
    create_result = JSON.parse(responce.body)
    cid = create_result["id"]
    sleep(2)

    #------------------show details------------------
    uri2 = URI.parse("http://api.paiza.io/runners/get_details?id=#{cid}&api_key=guest")
    responce = Net::HTTP.get_response(uri2)
    details_result = JSON.parse(responce.body)
    puts JSON.pretty_generate(details_result)
    if details_result["result"] == "success"
      text = "@#{screen_name}\n" + details_result["stdout"]
    elsif details_result["result"] == "failure"
      text = "@#{screen_name}\n" + details_result["stderr"]
    elsif details_result["result"].nil?
      text = "@#{screen_name}\n" + details_result["build_stderr"]
    else
      text = "@#{screen_name}\n" + "何らかのエラーが発生しました。"
    end
    if text.length > 140
      text = text[0...138] + "..."
    end
    @client.update text
  end
end

##ソースコードの取得・連結・再構築

実行したいコードを構成する関数は2つあります。

  • sourcejoinfirst
  • sourcejoin

言葉だけで説明するのがちょっと難しいので図とコードを載せます。簡単に言うと、「1つのツイートで完結するようなコード」はsourcejoinfirstだけでテキストを取得し、「何個のツイートにも及ぶコード」はsourcejoinfirstsourcejoinを使って再帰でテキストを連結してコードを構成すると言う仕組みです。

Untitled Diagram.jpg

sourcejoin.rb
require_relative '../Privatekey/oauth_twitter'
require_relative 'apifunc'
require 'json'
require 'oauth'

def sourcejoinfirst(tweet_id)
  info = show(tweet_id)
  if info["in_reply_to_status_id_str"].class == String
    final_info = show(info["in_reply_to_status_id_str"])
    source2 = final_info["full_text"]
    if final_info["in_reply_to_status_id_str"].class == String
      sourcejoin(source2, final_info["id_str"])
    else
      return source2
    end
  else
    return "error[NO TEXT]"
  end
end

def sourcejoin(text, id_str)
  source = text
  current_id = id_str
  current_info = show(current_id)

  if current_info["in_reply_to_status_id_str"].class == String
    before_id = current_info["in_reply_to_status_id_str"]
    before_info = show(before_id)
    source = before_info["full_text"] + "\n" + source
    if before_info["in_reply_to_status_id_str"].class == String
      sourcejoin(source, before_id)
    else
      return source
    end
  end
end

##その他やったこと
他にも、

  • Twitterから取得したテキストは&,<,>が文字実体参照として表示されるのでそれを直すメソッド
  • TwitterAPIのstatuses/show(id)を頻繁に使うのでちょっと簡単にしたメソッド

などを作りました。(大したものではないので省略)

#課題

今のところ見つけた不具合や直さなきゃいけないところです。テストはRubyでしかやってないのでまだまだ課題点は沢山あると思いますが…。

  • インスタンス変数を使うときが不便。Rubyは、インスタンス変数は@から始めるのがルールですが、Twitterだと誰かに対するリプライになってしまう場合があります。それだけならまだいいのですが(よくはない)そのまま自分にリプライしてコードを書き続けていると、取得した後のテキストの先頭にいちいちユーザーIDが入ってしまいます。いわゆる「巻き込みリプ」がそのまま残ってしまう現象です。ツイートを行う際にメンションを外すことで回避できるはずなのですが、ちょっと面倒かなって思います。

  • Twitterの自動URL短縮機能によって(適当な文字列).(拡張子)hoge.newが勝手にリンクになってしまう。これを直す方法は存在しないため、頭を抱えているところです。Twitterこそクソアプリなのかもしれない 少し確認したところ拡張子だけでも.py.javaをつけると文字が青くなり自動的にリンクになってしまいます。指定する拡張子ならこのアプリ独自のものを使用すればいいのですが(→そうしました。GithubのREADMEを参照)、コード内で青くなるような表現を探し出していちいち処理を書くのは大変そう…。コードを書くときは、文字が青くならないようにしましょうね、と言うしかないようです。

  • もしこのプログラムを他の人が実行していたとして、正規表現で実行コマンドを拾っていたらいろんな人からリプライが来るのでは問題。(他の人に使われると言うことを考えていなかった…)

#最後に
まだまだ未熟な技術力ですが、やっぱり自分でこれ作りたい!って思ってモノ作るのって楽しいですね。

色んなことを知りたいので、気軽にコメントなどしていただけると嬉しいです!

こんなものも作ってます↓
【TwitterAPI】「贅沢な名だねえ。今からお前の名前は○○だ。」【Ruby】

自分のツイートを感情分析して機嫌を管理する

#参考にしたもの

21
8
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
21
8

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?