Generate Speech Synthesis Markup Language (SSML) for Amazon Alexa with .NET.
Alexa Speech
for .NET is a library that will help you generate valid SSML for use with Amazon Alexa. Instead of using string concatenation or worring when using special characters you can use this library to avoid and eliminate these problems with the provided clean and easy to use fluent API.
This library is based on the Fluent-ish SSML Generator library which generates SSML as per Speech Synthesis Markup Language (SSML) Version 1.1 but it is optimized for usage with Amazon Alexa.
The library supports the following SSML tags:
- amazon:effect
- audio
- break
- emphasis
- p
- phoneme
- prosody
- s
- say-as - supports the following interpret-as values and formats as per Amazon Alexa Speech Synthesis Markup Language (SSML) Reference
- characters
- spell-out
- cardinal
- number
- ordinal
- digits
- fraction
- unit
- date
- time
- telephone
- address
- interjection
- expletive
- speak
- sub
- w - note that the previous tags used from the
ivona
namespace in the attribute names are not supported- amazon:VB
- amazon:VBD
- amazon:NN
- amazon:SENSE_1
Alexa Speech
can be installed via package manager console by executing the following command:
PM> Install-Package Alexa.Speech
or by using the dotnet CLI:
$ dotnet add package Alexa.Speech
Following are examples of how to use Alexa Speech
to generate valid SSML with the supported tags for your Alexa Skills:
string speech = new Speech()
.Say("I want to tell you a secret.")
.Say("I am not a real human.")
.WithEffect(AmazonEffect.Whisper)
.Say("Can you believe it?")
.Build();
<speak>
I want to tell you a secret.
<amazon:effect name="whispered">I am not a real human.</amazon:effect>
Can you believe it?
</speak>
string speech = new Speech()
.Say("Welcome to Car-Fu.")
.Play("https://s3.amazonaws.com/ask-soundlibrary/transportation/amzn_sfx_car_accelerate_01.mp3")
.Say("You can order a ride, or request a fare estimate. Which will it be?")
.Build();
<speak>
Welcome to Car-Fu.
<audio src="https://s3.amazonaws.com/ask-soundlibrary/transportation/amzn_sfx_car_accelerate_01.mp3" />
You can order a ride, or request a fare estimate.
Which will it be?
</speak>
string speech = new Speech()
.Say("There will be 3 seconds pause here")
.Pause()
.For(TimeSpan.FromSeconds(3))
.Say("then the speech continues.")
.Build();
<speak>
There will be 3 seconds pause here
<break time="3000ms" />
then the speech continues.
</speak>
string speech = new Speech()
.Say("I already told you I")
.Say("really like")
.Emphasise(EmphasiseLevel.Strong)
.Say("that person.")
.Build();
<speak>
I already told you I
<emphasis level="strong">really like</emphasis>
that person.
</speak>
string speech = new Speech()
.Say("This is the first paragraph. There should be a pause after this text is spoken.")
.AsParagraph()
.Say("This is the second paragraph.")
.AsParagraph()
.Build();
<speak>
<p>This is the first paragraph. There should be a pause after this text is spoken.</p>
<p>This is the second paragraph.</p>
</speak>
string speech = new Speech()
.Say("You say,")
.Say("pecan")
.AsPhoneme(PhoneticAlphabet.IPA, "pɪˈkɑːn")
.Say("but I say,")
.Say("pecan")
.AsPhoneme(PhoneticAlphabet.IPA, "ˈpi.kæn")
.Build();
<speak>
You say,
<phoneme alphabet="ipa" ph="pɪˈkɑːn">pecan</phoneme>
but I say,
<phoneme alphabet="ipa" ph="ˈpi.kæn">pecan</phoneme>
</speak>
string speech = new Speech()
.Say("Normal volume for the first sentence.")
.Say("Louder volume for the second sentence.")
.WithVolume(SpeechVolume.ExtraLoud)
.Say("When I wake up,")
.Say("I speak quite slowly.")
.WithRate(SpeechRate.ExtraSlow)
.Say("I can speak with my normal pitch,")
.Say("but also with a much higher pitch,")
.WithPitch(SpeechPitch.ExtraHigh)
.Say("and also")
.Say("with a lower pitch.")
.WithPitch(SpeechPitch.Low)
.Build();
<speak>
Normal volume for the first sentence.
<prosody volume="x-loud">Louder volume for the second sentence.</prosody>
When I wake up, <prosody rate="x-slow">I speak quite slowly.</prosody>
I can speak with my normal pitch, <prosody pitch="x-high">but also with a much higher pitch,</prosody>
and also <prosody pitch="low">with a lower pitch.</prosody>
</speak>
string speech = new Speech()
.Say("This is a sentence")
.AsSentence()
.Say("There should be a short pause before this second sentence")
.AsSentence()
.Say("This sentence ends with a period and should have the same pause.")
.Build();
<speak>
<s>This is a sentence</s>
<s>There should be a short pause before this second sentence</s>
This sentence ends with a period and should have the same pause.
</speak>
string speech = new Speech()
.Say(251)
.AsCardinal()
.Say("spoken as a cardinal number.")
.Say(251)
.AsOrdinal()
.Say("spoken as an ordinal number.")
.Say(1098468341)
.AsDigits()
.Say("is how you spell each digit separately.")
.Say("3/20")
.AsFraction()
.Say("is one common fraction.")
.Say("1kg")
.AsUnit()
.Say("is how much apples I will buy.")
.Say("100 John St.")
.AsAddress()
.Say("is the address where I live.")
.Say("I think that this comment is kind of")
.Say("ignorant.")
.Expletive()
.Say("1'47''")
.AsTime()
.Say("was the time I needed to write this sample.")
.Say("Call")
.Say("+46 (0)70 123 4567")
.AsTelephone()
.Say("to contact me.")
.Say("abcdefg")
.AsCharacters()
.Say("this is how I spell.")
.Say("gfedcba")
.SpellOut()
.Say("another way to do it.")
.Say(new DateTime(2018, 3, 15))
.Say("is today's date.")
.Say("Wow,")
.AsInterjection()
.Say("this was a long sample.")
.Build();
<speak>
<say-as interpret-as="cardinal">251</say-as> spoken as a cardinal number.
<say-as interpret-as="ordinal">251</say-as> spoken as an ordinal number.
<say-as interpret-as="digits">1098468341</say-as> is how you spell each digit separately.
<say-as interpret-as="fraction">3/20</say-as> is one common fraction.
<say-as interpret-as="unit">1kg</say-as> is how much apples I will buy.
<say-as interpret-as="address">100 John St.</say-as> is the address where I live.
I think that this comment is kind of <say-as interpret-as="expletive">ignorant.</say-as>
<say-as interpret-as="time">1'47''</say-as> was the time I needed to write this sample.
Call <say-as interpret-as="telephone">+46 (0)70 123 4567</say-as> to contact me.
<say-as interpret-as="characters">abcdefg</say-as> this is how I spell.
<say-as interpret-as="spell-out">gfedcba</say-as> another way to do it.
<say-as interpret-as="date">20180315</say-as> is today's date.
<say-as interpret-as="interjection">Wow,</say-as> this was a long sample.
</speak>
string speech = new Speech()
.Say("This is what Alexa sounds like without any SSML.")
.Build();
<speak>
This is what Alexa sounds like without any SSML.
</speak>
sub
string speech = new Speech()
.Say("My favorite chemical element is")
.Say("Al")
.AsAlias("aluminum")
.Say("but Al prefers")
.Say("Mg")
.AsAlias("magnesium")
.Say(".")
.Build();
<speak>
My favorite chemical element is <sub alias="aluminum">Al</sub>
but Al prefers <sub alias="magnesium">Mg</sub>.
</speak>
sub
string speech = new Speech()
.Say("The word")
.Say("read")
.AsCharacters()
.Say("may be interpreted as either the present simple form")
.Say("read")
.PronounceAs(PronounceRole.Verb)
.Say("or the past participle form")
.Say("read")
.PronounceAs(PronounceRole.PastParticiple)
.Build();
<speak>
The word <say-as interpret-as="characters">read</say-as> may be interpreted as either the present simple form
<w role="amazon:VB">read</w> or the past participle form <w role="amazon:VBD">read</w>
</speak>
sub
You can checkout all possible ways of using this library within the Alexa.Speech.Tests.csproj
project.
You are free to contribute to this project, just follow these simple rules:
- Submit an issue describing your proposed change.
- Fork this repo and develop.
- Cover your changes with Unit Tests and make sure they all pass.
- Make sure that your code adheres to the existing coding style.
- Submit a pull request.