RSSフィード読上げ Emic2
2023.09.16
まずは上記タイトル画像をクリックして、YouTube動画を再生してみてください。
今回は、ニュースサイトのRSSフィードを読み込んで、ニュースのヘッドラインを読み上げます。
●テキストスピーチ モジュール(Emic2)
英文(+スペイン語)テキストを音声出力するモジュールです。
TTLレベルRS232C信号で、テキストを送るだけで簡単に英語でしゃべります。
【特長】
・英語とスペイン語の高品質な音声合成
・男性、女性、子ども等9種のボイススタイル
・ピッチ、話す速度、ワード強調の音声特性の動的制御ができます。
・日本語をローマ字で送ると、外国人的な日本語発音でしゃべります。
・アンプ内蔵でスピーカ(付属していません)に直結できます。
【主な仕様】
・電源:+5V 30mA(アイドリング時)
46~220mA(スピーカ出力時)
・通信速度 :9600bps
・動作温度範囲:-20℃~+70℃
こちらの商品は秋月電子通商から購入しました。すでに生産終了している商品ですが2023年9月時点(5980円)でもまだ在庫があります。市場では2万円以上の高値で販売しているショップもあります。
●Pinout
使い方はとても簡単で、Emic2とマイコンボードをシリアル接続します。
●Emic2コマンド
起動時、Emic2に改行コード("\n")を送信し、Emic2からコロン(":")が返信されるのを待ちます。
Emic2からコロン(":")が送信された状態がコマンド入力モードです。
各コマンドは、CR あるいは LF を終端コードとして送信します。
Sx |
Convert text-to-speech
x = message
1023 characters maximum)
|
Dx |
Play demonstration message:
x = 0: English Introduction
x = 1: Singing "Daisy Bell"
x = 2: Spanish Introduction
|
Nx |
Select voice:
x = 0: Perfect Paul (Paulo)
x = 1: Huge Harry (Francisco)
x = 2: Beautiful Betty
x = 3: Uppity Ursula
x = 4: Doctor Dennis (Enrique)
x = 5: Kit the Kid
x = 6: Frail Frank
x = 7: Rough Rita
x = 8: Whispering Wendy (Beatriz)
|
Vx |
Set audio volume(dB):
x = -48 to 18
Upon power-up, the default value is 0.
|
Wx |
Set speaking rate (words/minute):
x = 75(slowest) to 600(fastest).
Upon power-up, the default value is 200.
|
Lx |
Select language:
x = 0 (US English)
x = 1 (Castilian Spanish)
x = 2 (Latin Spanish)
Upon power-up, the default value is 0.
|
Px |
Select parser:
x = 0 (DECtalk)
x = 1 (Epson)
Upon power-up, the default value is 1.
|
R |
Revert to default text-to-speech settings
Reset the user-configurable text-to-speech settings to their default, power-up configuration:
Voice = 0 (Paul)
Volume = 0 dB
Rate = 200 words/minute
Language = 0 (US English)
Parser = 1 (Epson)
|
C |
Print current text-to-speech settings
Displays the current values to the user-configurable text-to-speech settings.
|
I |
Print version information
|
H |
Print list of available commands
|
●Adafruit QT Py ESP32-S2 WiFi Dev Board with STEMMA QT
このマイコンボードは、WiFi機能と2MBのPSRAMを実装しています。このボードからUART接続によりEmic2を制御します。
・ESP32-S2 240MHz
・4 MB Flash & 2 MB PSRAM
・2.4 GHz Wi-Fi (SoC)
・Two I2C ports
・Hardware UART
・Hardware SPI
・Hardware I2S on any pins
・3.3V regulator with 600mA peak output
Adafruit QT Py ESP32-S2
●Qt Py ESP32-S2 Pinout
●配線
左が Emic2、中央のスピーカの背後にあるアンプで音声を増幅しています。右は Adafruit QT Py ESP32-S2 です。
SPEAKER 8Ω | - | Emic2 | - | ESP32S2 |
| | GND | - | GND |
| | 5V | - | 3V |
| | SOUT | - | RX |
| | SIN | - | TX |
SPEAKER(ー) | - | SPー | | |
SPEAKER(+) | - | SP+ | | |
●開発環境
ソースコードのビルドには、PlatformIOを使用しています。
Arduino開発環境構築 PlatformIO
環境設定を行います。
$ mkdir -p ~/Emic2
$ cd ~/Emic2
$ pio init -b adafruit_qtpy_esp32s2
$ vi platformio.ini
[env:adafruit_qtpy_esp32s2]
platform = espressif32
board = adafruit_qtpy_esp32s2
framework = arduino
platform_packages =
framework-arduinoespressif32 @ https://github.com/espressif/arduino-esp32
build_flags =
-DBOARD_HAS_PSRAM
-mfix-esp32-psram-cache-issue
upload_port = /dev/ttyACM0
●Emic2 gitHub
nlamprian/EMIC2
Emic2用ライブラリですが、使用していません。プログラム作成の参考にしている程度です。
●Emic2 動作確認用コード
void setup()
{
Serial1.begin(9600);
Serial1.print('\n');
while (Serial1.read() != ':');
delay(10);
Serial1.flush();
// Set audio volume(dB)
Serial1.print("V-10");
Serial1.print("\n");
while (Serial1.read() != ':');
delay(10);
Serial1.flush();
// Set speaking rate (words/minute)
Serial1.print("W200");
Serial1.print("\n");
while (Serial1.read() != ':');
delay(10);
Serial1.flush();
// Select voice
Serial1.print("N0");
Serial1.print("\n");
while (Serial1.read() != ':');
delay(10);
Serial1.flush();
Serial1.print("S");
Serial1.print("Hello. My name is the Emic 2 Text-to-Speech module.");
Serial1.print("\n");
while (Serial1.read() != ':');
delay(10);
Serial1.flush();
}
void loop() {}
Emic2 とのシリアル通信速度は 9600bps に設定します。
初期設定では、音量、読み上げ速度、声の種類を指定します。
テキストの読み上げにはSコマンドに続いて、テキストを送信します。
●BBC RSS
実際にBBC(英国放送協会)のRSSフィードを取得して読み上げます。
News feeds from the BBC
RSSの仕組みについては下記を参考にしてください。
XML Parser (ESP32/Arduino)
ニュースサイトのRSSフィードのXML階層構造は下記のようになっています。
この中の <description> タグで囲まれているヘッドライン情報を抽出して、Emic2 に流し込みます。
●BBC News feeds 抽出 & 読上げコード
認証情報ファイル datalink.h
#define SECRET_SSID "XXXXXXXXXXXX"
#define SECRET_PASS "XXXXXXXXXXXXX"
const char* bbc_co_uk_pem = \
"-----BEGIN CERTIFICATE-----\n" \
"MIIDXzCCAkegAwIBAgILBAAAAAABIVhTCKIwDQYJKoZIhvcNAQELBQAwTDEgMB4G" \
"A1UECxMXR2xvYmFsU2lnbiBSb290IENBIC0gUjMxEzARBgNVBAoTCkdsb2JhbFNp" \
"Z24xEzARBgNVBAMTCkdsb2JhbFNpZ24wHhcNMDkwMzE4MTAwMDAwWhcNMjkwMzE4" \
"MTAwMDAwWjBMMSAwHgYDVQQLExdHbG9iYWxTaWduIFJvb3QgQ0EgLSBSMzETMBEG" \
"A1UEChMKR2xvYmFsU2lnbjETMBEGA1UEAxMKR2xvYmFsU2lnbjCCASIwDQYJKoZI" \
"hvcNAQEBBQADggEPADCCAQoCggEBAMwldpB5BngiFvXAg7aEyiie/QV2EcWtiHL8" \
"RgJDx7KKnQRfJMsuS+FggkbhUqsMgUdwbN1k0ev1LKMPgj0MK66X17YUhhB5uzsT" \
"gHeMCOFJ0mpiLx9e+pZo34knlTifBtc+ycsmWQ1z3rDI6SYOgxXG71uL0gRgykmm" \
"KPZpO/bLyCiR5Z2KYVc3rHQU3HTgOu5yLy6c+9C7v/U9AOEGM+iCK65TpjoWc4zd" \
"QQ4gOsC0p6Hpsk+QLjJg6VfLuQSSaGjlOCZgdbKfd/+RFO+uIEn8rUAVSNECMWEZ" \
"XriX7613t2Saer9fwRPvm2L7DWzgVGkWqQPabumDk3F2xmmFghcCAwEAAaNCMEAw" \
"DgYDVR0PAQH/BAQDAgEGMA8GA1UdEwEB/wQFMAMBAf8wHQYDVR0OBBYEFI/wS3+o" \
"LkUkrk1Q+mOai97i3Ru8MA0GCSqGSIb3DQEBCwUAA4IBAQBLQNvAUKr+yAzv95ZU" \
"RUm7lgAJQayzE4aGKAczymvmdLm6AC2upArT9fHxD4q/c2dKg8dEe3jgr25sbwMp" \
"jjM5RcOO5LlXbKr8EpbsU8Yt5CRsuZRj+9xTaGdWPoO4zzUhw8lo/s7awlOqzJCK" \
"6fBdRoyV3XpYKBovHd7NADdBj+1EbddTKJd+82cEHhXXipa0095MJ6RMG3NzdvQX" \
"mcIfeg7jLQitChws/zyrVQ4PkX4268NXSb7hLi18YIvDQVETI53O9zJrlAGomecs" \
"Mx86OyXShkDOOyyGeMlhLxS67ttVb9+E7gUJTb0o2HLO02JQZR7rkpeDMdmztcpH" \
"WD9f" \
"-----END CERTIFICATE-----";
サンプルコード
#include <Wire.h>
#include <WiFi.h>
#include <WiFiClientSecure.h>
#include <NunniMCAX.h>
#include <datalink.h>
#define SLAVE_ADDR 0x77
WiFiClient http_client;
WiFiClientSecure https_client;
#define MAXLEN 256
static char m_characters[MAXLEN];
struct NunniMCAXContentHandler handler;
bool RootDetected = false;
typedef struct {
char domain[32];
char page[32];
int port;
} URL_INFO;
URL_INFO url;
#define XML_BUF_SIZE (32768)
typedef struct {
long length;
long pos;
char *buf;
} XML_BUF;
XML_BUF xml;
#define I2C_BUF_SIZE (32)
void URLAnalyse(char *str)
{
char *inpt, *outpt;
if (strncmp(str,"http://",7)==0) {
url.port = 80;
inpt = str + 7;
} else {
url.port = 443;
inpt = str + 8;
}
for (outpt = url.domain; (*inpt!='/')&&(*inpt!=(char)NULL); outpt++,inpt++) *outpt = *inpt;
url.page[0] = *inpt;
*outpt = (char)NULL;
for (outpt = &url.page[1],inpt++; *inpt!=(char)NULL; outpt++,inpt++) *outpt = *inpt;
*outpt = (char)NULL;
}
void getRSS(const char *pem)
{
char get_cmd[256];
unsigned long timeout = millis();
if (url.port == 443) {
https_client.setCACert(pem);
if (https_client.connect(url.domain, url.port)) {
sprintf(get_cmd, "GET https://%s%s HTTP/1.1\r\nHost: %s\r\nUser-Agent: BuildFailureDetectorESP32\r\nConnection: close\r\n\r\n\0", url.domain, url.page, url.domain);
https_client.print(get_cmd);
https_client.flush();
while(https_client.available() == 0){
if(millis() - timeout > 5000){
https_client.stop();
return;
}
}
xml.pos = 0;
while (1) {
if (https_client.available() ) {
xml.buf[xml.pos] = https_client.read();
xml.pos++;
}
if (!https_client.connected()) {
xml.buf[xml.pos] = EOF;
xml.length = xml.pos;
xml.pos = -1;
break;
}
}
} else {
return;
}
} else {
if (http_client.connect(url.domain, url.port)) {
http_client.write("GET /rss/topics/domestic.xml HTTP/1.1\r\n\r\n");
http_client.flush();
while(http_client.available() == 0){
if(millis() - timeout > 5000){
http_client.stop();
return;
}
}
xml.pos = 0;
while (1) {
if (http_client.available() ) {
xml.buf[xml.pos] = http_client.read();
xml.pos++;
}
if (!http_client.connected()) {
xml.buf[xml.pos] = EOF;
xml.length = xml.pos;
xml.pos = -1;
break;
}
}
} else {
return;
}
}
// パーサーの呼び出し
NunniMCAXparse( &getcFunc, &handler );
if (url.port == 443) {
https_client.stop();
} else {
http_client.stop();
}
RootDetected = false;
delay(5000);
return;
}
// Ethernetからの1文字読み取るコールバック関数
// 最初の'<'(タグの開始)まで読み飛ばし
// → HTTPサーバーレスポンス文字を読み飛ばす
int getcFunc()
{
while(!RootDetected) {
if (getChar() == '<') {
RootDetected = true;
return '<';
}
}
return getChar();
}
int getChar()
{
xml.pos++;
return xml.buf[xml.pos];
}
/* SAXパーサーのイベントハンドラー */
int startDocument(void)
{
return 0;
}
int startElement( const char *tagname, struct NunniHashtable *args )
{
const int size = NunniHashtableSize( args );
char ** keys;
int i, ret;
const char *name, *value;
keys = (char**)calloc( size, sizeof( char * ) );
ret = NunniHashtableKeys( args, keys );
for ( i = 0; i < size; ++i ) {
name = keys[i];
value = NunniHashtableGet( args, name );
}
memset( m_characters, 0, MAXLEN );
free(keys);
return 0;
}
int characters( char ch[], int start, int length )
{
int i = strlen( m_characters );
if ( i == MAXLEN ) return -1;
strncat( m_characters, &(ch[start]), length );
return 0;
}
int endElement( const char *tagname )
{
int len;
char *data = m_characters;
while( isspace( *data ) ) {
++data;
}
len = strlen( data );
while( isspace( data[--len] ) ) data[len] = 0;
if ( data != NULL && strncmp( data, "", 1 ) ) {
if (strncmp(tagname,"description",11)==0) {
Emic2_speech(data);
delay(1000);
}
}
memset( m_characters, 0, MAXLEN );
return 0;
}
int endDocument(void)
{
return 0;
}
void Emic2_init()
{
Serial1.print('\n');
while (Serial1.read() != ':');
delay(10);
Serial1.flush();
// Set audio volume(dB)
Serial1.print("V-10");
Serial1.print("\n");
while (Serial1.read() != ':');
delay(10);
Serial1.flush();
// Set speaking rate (words/minute)
Serial1.print("W200");
Serial1.print("\n");
while (Serial1.read() != ':');
delay(10);
Serial1.flush();
// Select voice
Serial1.print("N0");
Serial1.print("\n");
while (Serial1.read() != ':');
delay(10);
Serial1.flush();
}
void Emic2_speech(char *text)
{
Serial1.print("S");
Serial1.print(text);
Serial1.print("\n");
while (Serial1.read() != ':');
delay(10);
Serial1.flush();
}
void setup()
{
Wire.begin();
Serial1.begin(9600);
WiFi.begin(SECRET_SSID, SECRET_PASS);
while (WiFi.status() != WL_CONNECTED) delay(500);
Emic2_init();
// set up the paser handler functions
handler.startDocument = startDocument;
handler.startElement = startElement;
handler.characters = characters;
handler.endElement = endElement;
handler.endDocument = endDocument;
xml.buf = (char*)malloc(XML_BUF_SIZE);
URLAnalyse("https://feeds.bbci.co.uk/news/world/rss.xml");
getRSS(bbc_co_uk_pem);
}
void loop() {}
以前紹介した XML Parser では、RSSサイトから逐次読み込んで、構造解析を行っていましたが、
Adafruit QT Py ESP32S2 は 2MB の PSRAM を実装しているので、まずは一気にRSSフィードを読み込んで、メモリ上に保存しています。
●応用
マイコンボード起動時にNTPサーバから時刻情報を取得して、特定の時刻に各種RSSフィードを読み込み、読み上げることが可能です、
FOXNews.com RSS Feeds New York, US
News feeds from the BBC London, England, UK
The New York Times RSS Feed New York, US
Al Jazeera RSS Feed Qatar
E-International Relations RSS Feed UK
Washington Post RSS feeds Washington, US
France 24 RSS Feed Paris, Ile-de-France, France
読み上げと同時にヘッドラインテキストをモニターに表示するのもよいかもしれません。
|
Raspberry Pi(ラズベリー パイ)は、ARMプロセッサを搭載したシングルボードコンピュータ。イギリスのラズベリーパイ財団によって開発されている。
Arduinoで学ぶ組込みシステム入門(第2版)
●Arduinoを使って組込みシステム開発を理解する
・ハードウェアやソフトウェアなどの基礎知識/
・設計から実装までを系統的に説明するモデルベース開発/
・Arduinoを用いた実際の開発例
最新 使える! MATLAB 第3版
◆◆すぐに「使える!」 全ページフルカラー!◆◆
・MATLAB R2022bに対応し、解説もより詳しく!/
・コマンド・スクリプトの例が豊富で、動かして学べる!/
・超基本から解説。これから使いはじめる人にぴったり!/
・全編フルカラー、スクリーンショットも豊富!
Amazon Web Services基礎からのネットワーク&サーバー構築改訂4版
1.システム構築をインフラから始めるには/
2.ネットワークを構築する/
3.サーバーを構築する/
4.Webサーバーソフトをインストールする/
5.HTTPの動きを確認する/
6.プライベートサブネットを構築する/
7.NATを構築する/
8.DBを用いたブログシステムの構築/
9.TCP/IPによる通信の仕組みを理解する
C言語は第二の母国語: 独学学生時代から企業内IT職人時代に培った、独立のための技術とノウハウ 平田豊著
学生時代から独学でプログラミングをはじめ、企業内でデバイスドライバを開発し、そして独立後もたくさんのアプリケーション開発や技術書制作に携わってきた著者。その筆者が大事に使い続ける「C言語」の“昔と今”について、気づいたことや役立つ知識、使ってきたツールなどについて、これまで記してきたことを整理してまとめました。
本書では、現役プログラマーだけでなく、これからプログラミングを学ぶ学生などにも有益な情報やノウハウを、筆者の経験を元に紹介しています。
1冊ですべて身につくJavaScript入門講座
・最初の一歩が踏み出せる! 初心者に寄り添うやさしい解説
・最新の技術が身につく! 今のJavaScriptの書き方・使い方
・絶対に知っておきたい! アニメーションとイベントの知識
・プログラミングの基本から実装方法まですべて学べる
図解! Git & GitHubのツボとコツがゼッタイにわかる本
ソフトウェア開発では欠かすことのできないGit、GitHub。
これからGit、GitHubを使いたいという入門者の方でも、実際に手を動かしながら使い方を学べます。
C自作の鉄則!2023 (日経BPパソコンベストムック)
メーカー製のパソコンはスペックが中途半端で、自分が本当に欲しい機種がない――。そう思っている人には、ぜひ自作パソコンをお薦めします。自作パソコンのパーツは進化が速く、しかも驚くほど種類が豊富。価格も性能も、幅広く用意されているため、満足度100%の“自分だけの1台”を手に入れることができます。
Interface 2023年6月号
特集:第1部 フィルタ設計 基礎の基礎/
第2部 係数アプリや波形観測アプリで合点!FIR&IIRフィルタ作り/
第3部 配布プリント基板で体験!マイコンで動くフィルタ作り
日経Linux 2023年5月号
【特集 1】 AI時代の最強フリーソフト ~ 25のやりたいを実現!
【特集 2】 AWS、Azureのうまみを無料で体感!面倒なことはクラウドに任せよう
【特集 3】 新しいRaspberry Pi Cameraで遊んでみよう
【特集 4】 Linuxで旧型PCを復活! 1kg切るモバイルPCを「ChromeOS Flex」でChromebook化
ラズパイマガジン2022年秋号
特集:5大人気ボード 電子工作超入門
「半導体不足で在庫が不足し、電子工作のボードがなかなか買えない…」。そんな今にふさわしい特集を企画しました。5種の人気ボードにすべて対応した電子工作の入門特集です。「GPIO」や「I2C」を使った電子パーツの制御方法は、どのボードでも同じです。手に入れられたボードを使って、今こそ電子工作を始めましょう。
地方で稼ぐ! ITエンジニアのすすめ
学歴、理系の知識、専門スキル……全部なくてもITエンジニアになれる!
地方でも高収入でやりがいをもって働ける!ITエンジニアの魅力を一挙大公開
Raspberry Piのはじめ方2022
本書は、ラズパイやPicoの買い方やインストール、初期設定といった基本から、サーバー、電子工作、IoT、AIといったラズパイならではの活用方法まで、1冊でお届けします。
ラズパイをこれから始める方向けに、全36ページの入門マンガ「女子高生とラズベリーパイ」も巻末に掲載。これを読むだけでラズパイがどんなものなのか、すぐに分かって触れるようになります。
ハッカーの学校 IoTハッキングの教科書
生活にとけこみ、家電機器を便利にするIoT技術。
Webカメラなど、便利の裏側に潜むセキュリティの危険性をハッキングで検証。
専門家がパケットキャプチャからハードウェアハッキングまで、その攻撃と防御を徹底解説。
本書は2018年7月に刊行された「ハッカーの学校IoTハッキングの教科書」に一部修正を加えた第2版です。
攻撃手法を学んで防御せよ! 押さえておくべきIoTハッキング
本書は、経済産業省から2021年4月にリリースされた、IoTセキュリティを対象とした『機器のサイバーセキュリティ確保のためのセキュリティ検証の手引き』の『別冊2 機器メーカに向けた脅威分析及びセキュリティ検証の解説書』をもとに、IoT機器の開発者や品質保証の担当者が、攻撃者の視点に立ってセキュリティ検証を実践するための手法を、事例とともに詳細に解説しました。
ポチらせる文章術
販売サイト・ネット広告・メルマガ・ブログ・ホームページ・SNS…
全WEB媒体で効果バツグン!
カリスマコピーライターが教える「見てもらう」「買ってもらう」「共感してもらう」すべてに効くネット文章術
プログラマーは世界をどう見ているのか 西村博之著
イーロン・マスク(テスラ)、ジェフ・べゾス(Amazon)、ラリー・ペイジ(Google)…etc.
世界のトップはなぜプログラマーなのか?
ニーア オートマタ PLAY ARTS改 <ヨルハ 二号 B型 DX版> PVC製 塗装済み可動フィギュア
「NieR:Automata」より、ヨルハ二号B型こと2BがPLAY ARTS改に新たに登場!
高級感の感じられるコスチュームや髪の質感、洗練されたボディバランス、細かなデティールに至るまでこだわり抜かれた逸品。
DX版には通常版のラインナップに加え2Bの随行支援ユニット ポッド042などをはじめ“純白の美しい太刀"白の約定やエフェクトパーツ、自爆モードを再現できる換装用ボディパーツ、シーンに合わせて変えられる顔パーツ2種も付属する豪華な仕様に。
作中のあらゆるシーンを再現することが可能なファン必見の一品となっている。
Newtonライト2.0 ベイズ統計
ベイズ統計は,結果から原因を推定する統計学です。AIや医療などの幅広い分野で応用されています。その基礎となるのは18世紀に考えだされた「ベイズの定理」です。
この本では,ベイズ統計学のきほんをやさしく紹介していきます。
白光(HAKKO) ダイヤル式温度制御はんだ吸取器 ハンディタイプ FR301-81
無水エタノールP 500mlx2個パック(掃除)
ケイバ(KEIBA) マイクロニッパー MN-A04
サンハヤト SAD-101 ニューブレッドボード
白光(HAKKO) HEXSOL 巻はんだ 精密プリント基板用 150g FS402-02
[Amazon限定ブランド]【指定第2類医薬品】PHARMA CHOICE 解熱鎮痛薬 解熱鎮痛錠IP 100錠
|