de.sciss.synth.Server.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of scalacollider_3 Show documentation
Show all versions of scalacollider_3 Show documentation
A sound synthesis library for the SuperCollider server
The newest version!
/*
* Server.scala
* (ScalaCollider)
*
* Copyright (c) 2008-2021 Hanns Holger Rutz. All rights reserved.
*
* This software is published under the GNU Affero General Public License v3+
*
*
* For further information, please contact Hanns Holger Rutz at
* [email protected]
*/
package de.sciss.synth
import de.sciss.audiofile.{AudioFileType, SampleFormat}
import de.sciss.model.Model
import de.sciss.numbers.{IntFunctions => ri}
import de.sciss.osc
import de.sciss.osc.{Browser, TCP, UDP}
import de.sciss.processor.Processor
import de.sciss.synth.impl.ServerImpl
import java.net.{DatagramSocket, InetAddress, ServerSocket}
import scala.collection.mutable
import scala.concurrent.Future
import scala.concurrent.duration._
import scala.language.implicitConversions
import scala.util.Try
object Server extends ServerPlatform {
def default: Server = ServerImpl.default
/** The base trait for `Config` and `ConfigBuilder` describes the settings used to boot scsynth in
* realtime or non-realtime mode, as well as its server address and port.
*
* You obtain a `ConfigBuilder` by calling `Server.Config()`. This builder can then be mutated and
* will be implicitly converted to an immutable `Config` when required.
*
* See `ConfigBuilder` for its default values.
*
* @see [[de.sciss.synth.Server.ConfigBuilder]]
* @see [[de.sciss.synth.Server.Config]]
*/
trait ConfigLike {
/** The path to `scsynth`, used when booting a server. This can be either a relative path
* (relating to the JVM's working directory), or an absolute path.
*
* @see [[de.sciss.synth.Server#defaultProgram]]
*/
def program: String
/** The maximum number of control bus channels. */
def controlBusChannels: Int
/** The maximum number of audio bus channels. This includes the channels connected
* to hardware (`outputBusChannels`) as well as all channels for internal routing.
*/
def audioBusChannels: Int
/** The number of connected audio hardware output channels. This does not need to
* correspond to the actual number of channels your sound card provides, but can
* be lower or higher, although a higher value doesn't have any effect as channel
* indices above the number of channels of the sound card will be treated as
* internal channels.
*/
def outputBusChannels: Int
/** The calculation block size. That is, the number of audio samples calculated en-bloc.
* This corresponds with the control rate, such that
* `controlRate := audioRate / blockSize`. It should be a power of two.
*/
def blockSize: Int
/** The audio hardware sampling rate to use. A value of `0` indicates that scsynth
* should use the current sampling rate of the audio hardware. An explicit setting
* will make scsynth try to switch the sound card's sample rate if necessary.
*/
def sampleRate: Int
/** The maximum number of audio buffers (for the `Buffer` class). */
def audioBuffers: Int
/** The maximum number of concurrent nodes (synths and groups). */
def maxNodes: Int
/** The maximum number of synth defs. */
def maxSynthDefs: Int
/** The maximum number of pre-allocated realtime memory in bytes. This memory
* is used for many UGens such as `Limiter`, `DelayN` etc. It does not
* affect dynamically allocated memory such as audio buffers.
*/
def memorySize: Int
/** The maximum number of concurrent connections between UGens in a single synth.
* ScalaCollider performs a depth-first topological sorting of the synth defs,
* so you should not worry too much about this value. It can become important
* in very heavy channel expansions and mix-down.
*
* This value will be automatically increased if a more complex def is loaded
* at startup, but it cannot be increased thereafter without rebooting.
*/
def wireBuffers: Int
/** The number of individual random number generators allocated. */
def randomSeeds: Int
/** Whether scsynth should load synth definitions stored on the hard-disk when booted. */
def loadSynthDefs: Boolean
/** ? */
def machPortName: Option[(String, String)]
/** The verbosity level of scsynth. The standard value is `0`, while
* `-1` suppresses informational messages, `-2` also suppresses many error messages.
*/
def verbosity: Int
/** An explicit list of paths where DSP plugins are found. Usually this is not
* specified, and scsynth looks for plugins in their default location.
*/
def plugInsPaths: List[String]
/** An option to restrict access to files (e.g. for loading and saving buffers) to
* a particular directory. This is a security measure, preventing malicious clients from
* accessing parts of the hard-disk which they shouldn't.
*/
def restrictedPath: Option[String]
// ---- realtime only ----
/** (Realtime) Host address of scsynth, when trying to `connect` to an already running server on the net. */
def host: String
/** (Realtime) UDP or TCP port used by scsynth. */
def port: Int
/** (Realtime) Open Sound Control transport used by scsynth. (Either of `UDP` and `TCP`). */
def transport: osc.Transport
/** (Realtime) An option to enable particular input 'streams' or 'bundles' of a sound card.
* This is a 'binary' String made of `'0'` and `'1'` characters.
* If the string is `"01100"`, for example, then only the second and third input streams on
* the device will be enabled.
*/
def inputStreamsEnabled: Option[String]
/** (Realtime) An option to enable particular output 'streams' or 'bundles' of a sound card.
* This is a 'binary' String made of `'0'` and `'1'` characters.
* If the string is `"01100"`, for example, then only the second and third output streams on
* the device will be enabled.
*/
def outputStreamsEnabled: Option[String]
/** (Realtime) An option denoting the name of the sound card to use. On systems which distinguish
* input and output devices (OS X), this implies that both are the same. Otherwise, you can
* use the `deviceNames` method instead.
*
* @see deviceNames
*/
def deviceName: Option[String]
/** (Realtime) An option denoting the name of the input and output sound device to use. This is for
* systems which distinguish input and output devices (OS X). If you use a single device both for
* input and output (applies to most professional audio interfaces), you can simply use the
* single string method `deviceName`.
*
* @see deviceName
*/
def deviceNames: Option[(String, String)]
/** (Realtime) The number of connected audio hardware input channels. This does not need to
* correspond to the actual number of channels your sound card provides, but can
* be lower or higher, although a higher value doesn't have any effect as channel
* indices above the number of channels of the sound card will be treated as
* internal channels.
*/
def inputBusChannels: Int
/** (Realtime) A value to adjust the sound card's hardware block size. Typically you will leave
* this to `0` which means that the current block size is used. The block sizes supported depend
* on the particular sound card. Lower values decrease latency but may increase CPU load.
*/
def hardwareBlockSize: Int
/** (Realtime) Whether to announce scsynth's OSC service via zero conf. See
* [[http://en.wikipedia.org/wiki/Zero_configuration_networking Wikipedia]] for more details.
*/
def zeroConf: Boolean
/** (Realtime) The maximum number of client connections when using TCP transport. */
def maxLogins: Int
/** (Realtime) A requires session password when using TCP transport. When using TCP and the password option
* is set, each client must send the correct password as the first command to the server, otherwise it is
* rejected.
*/
def sessionPassword: Option[String]
// ---- non-realtime only ----
/** (Non-Realtime) Path to the binary OSC file. */
def nrtCommandPath: String
/** (Non-Realtime) Path to the audio input file used as audio input bus supplement. */
def nrtInputPath: Option[String]
/** (Non-Realtime) Path to the audio output file used as audio output bus supplement. */
def nrtOutputPath: String
/** (Non-Realtime) Audio file format for writing the output. */
def nrtHeaderFormat: AudioFileType
/** (Non-Realtime) Audio sample format for writing the output. */
def nrtSampleFormat: SampleFormat
/** Produces a command line for booting scsynth in realtime mode. */
final def toRealtimeArgs: List[String] = Config.toRealtimeArgs(this)
/** Produces a command line for booting scsynth in non-realtime mode. */
final def toNonRealtimeArgs: List[String] = Config.toNonRealtimeArgs(this)
/** A utility method providing the audio bus offset for the start of
* the internal channels. (simply the sum of `outputBusChannels` and `inputBusChannels`).
*/
final def internalBusIndex: Int = outputBusChannels + inputBusChannels
}
object Config {
/** Creates a new configuration builder with default settings */
def apply(): ConfigBuilder = new ConfigBuilder()
/** Implicit conversion which allows you to use a `ConfigBuilder`
* wherever a `Config` is required.
*/
implicit def build(cb: ConfigBuilder): Config = cb.build
private[Server] def toNonRealtimeArgs(o: ConfigLike): List[String] = {
val b = List.newBuilder[String]
// -N <...other scsynth arguments>
b += o.program
b += "-N"
b += o.nrtCommandPath
b += o.nrtInputPath.getOrElse("_")
b += o.nrtOutputPath
b += o.sampleRate.toString
b += o.nrtHeaderFormat.id
b += o.nrtSampleFormat.id
addCommonArgs( o, b )
b.result()
}
private[Server] def toRealtimeArgs(o: ConfigLike): List[String] = {
val b = List.newBuilder[String]
b += o.program
o.transport match {
case TCP =>
b += "-t"
b += o.port.toString
case UDP | Browser =>
b += "-u"
b += o.port.toString
}
if (o.host != "0.0.0.0") {
b += "-B"
b += o.host
}
addCommonArgs(o, b)
if (o.hardwareBlockSize != 0) {
b += "-Z"
b += o.hardwareBlockSize.toString
}
if (o.sampleRate != 0) {
b += "-S"
b += o.sampleRate.toString
}
if (o.maxLogins != 64) {
b += "-l"
b += o.maxLogins.toString
}
o.sessionPassword.foreach { pwd =>
b += "-p"
b += pwd
}
o.inputStreamsEnabled.foreach { stream =>
b += "-I"
b += stream
}
o.outputStreamsEnabled.foreach { stream =>
b += "-O"
b += stream
}
if (!o.zeroConf) {
b += "-R"
b += "0"
}
o.deviceNames.foreach { case (inDev, outDev) =>
b += "-H"
b += inDev
b += outDev
}
o.deviceName.foreach { n =>
b += "-H"
b += n
}
o.restrictedPath.foreach { path =>
b += "-P"
b += path
}
b.result()
}
private[Server] def addCommonArgs(o: ConfigLike, b: mutable.Builder[String, Any]): Unit = {
// some dude is going around changing scsynth
// defaults without thinking about the consequences.
// we now pessimistically pass all options and do
// not assume any longer defaults.
// if (o.controlBusChannels != 4096) {
b += "-c"
b += o.controlBusChannels.toString
// }
// if (o.audioBusChannels != 128) {
b += "-a"
b += o.audioBusChannels.toString
// }
// if (o.inputBusChannels != 8) {
b += "-i"
b += o.inputBusChannels.toString
// }
// if (o.outputBusChannels != 8) {
b += "-o"
b += o.outputBusChannels.toString
// }
// if (o.blockSize != 64) {
b += "-z"
b += o.blockSize.toString
// }
// if (o.audioBuffers != 1024) {
b += "-b"
b += o.audioBuffers.toString
// }
// if (o.maxNodes != 1024) {
b += "-n"
b += o.maxNodes.toString
// }
// if (o.maxSynthDefs != 1024) {
b += "-d"
b += o.maxSynthDefs.toString
// }
// if (o.memorySize != 8192) {
b += "-m"
b += o.memorySize.toString
// }
// if (o.wireBuffers != 64) {
b += "-w"
b += o.wireBuffers.toString
// }
// if (o.randomSeeds != 64) {
b += "-r"
b += o.randomSeeds.toString
// }
if (!o.loadSynthDefs) {
b += "-D"
b += "0"
}
o.machPortName.foreach {
case (send, reply) =>
b += "-M"
b += send
b += reply
}
if (o.verbosity != 0) {
b += "-V"
b += o.verbosity.toString
}
if (o.plugInsPaths.nonEmpty) {
b += "-U"
b += o.plugInsPaths.mkString(":")
}
}
}
/** @see [[de.sciss.synth.Server.ConfigBuilder]]
* @see [[de.sciss.synth.Server.ConfigLike]]
*/
final class Config private[Server](val program: String,
val controlBusChannels: Int,
val audioBusChannels: Int,
val outputBusChannels: Int,
val blockSize: Int,
val sampleRate: Int,
val audioBuffers: Int,
val maxNodes: Int,
val maxSynthDefs: Int,
val memorySize: Int,
val wireBuffers: Int,
val randomSeeds: Int,
val loadSynthDefs: Boolean,
val machPortName: Option[(String, String)],
val verbosity: Int,
val plugInsPaths: List[String],
val restrictedPath: Option[String],
/* val memoryLocking: Boolean, */
val host: String,
val port: Int,
val transport: osc.Transport,
val inputStreamsEnabled: Option[String],
val outputStreamsEnabled: Option[String],
val deviceNames: Option[(String, String)],
val deviceName: Option[String],
val inputBusChannels: Int,
val hardwareBlockSize: Int,
val zeroConf: Boolean,
val maxLogins: Int,
val sessionPassword: Option[String],
val nrtCommandPath: String,
val nrtInputPath: Option[String],
val nrtOutputPath: String,
val nrtHeaderFormat: AudioFileType,
val nrtSampleFormat: SampleFormat)
extends ConfigLike {
override def toString = "Server.Config"
}
object ConfigBuilder {
def apply(config: Config): ConfigBuilder = {
val b = new ConfigBuilder
b.read(config)
b
}
}
/** @see [[de.sciss.synth.Server.Config]]
* @see [[de.sciss.synth.Server.ConfigLike]]
*/
final class ConfigBuilder private[Server]() extends ConfigLike {
/** The default `program` is read from `defaultProgram`
*
* @see [[de.sciss.synth.Server#defaultProgram]]
*/
var program: String = defaultProgram
private[this] var controlBusChannelsVar = 4096
/** The default number of control bus channels is `4096` (scsynth default).
* Must be greater than zero and a power of two.
*/
def controlBusChannels: Int = controlBusChannelsVar
/** The default number of control bus channels is `4096` (scsynth default).
* Must be greater than zero and a power of two.
*/
def controlBusChannels_=(value: Int): Unit = {
require (value > 0 && ri.isPowerOfTwo(value))
controlBusChannelsVar = value
}
private[this] var audioBusChannelsVar = 128
/** The default number of audio bus channels is `128` (scsynth default).
* Must be greater than zero and a power of two.
* When the builder is converted to a `Config`, this value may be increased
* to ensure that `audioBusChannels > inputBusChannels + outputBusChannels`.
*/
def audioBusChannels: Int = audioBusChannelsVar
/** The default number of audio bus channels is `128` (scsynth default).
* Must be greater than zero and a power of two.
* When the builder is converted to a `Config`, this value may be increased
* to ensure that `audioBusChannels > inputBusChannels + outputBusChannels`.
*/
def audioBusChannels_=(value: Int): Unit = {
require (value > 0 && ri.isPowerOfTwo(value))
audioBusChannelsVar = value
}
private[this] var outputBusChannelsVar = 8
/** The default number of output bus channels is `8` (scsynth default) */
def outputBusChannels: Int = outputBusChannelsVar
/** The default number of output bus channels is `8` (scsynth default) */
def outputBusChannels_=(value: Int): Unit = {
require (value >= 0)
outputBusChannelsVar = value
}
private[this] var blockSizeVar = 64
/** The default calculation block size is `64` (scsynth default).
* Must be greater than zero and a power of two.
*/
def blockSize: Int = blockSizeVar
/** The default calculation block size is `64` (scsynth default).
* Must be greater than zero and a power of two.
*/
def blockSize_=(value: Int): Unit = {
require (value > 0 && ri.isPowerOfTwo(value))
blockSizeVar = value
}
private[this] var sampleRateVar = 0
/** The default sample rate is `0` (meaning that it is adjusted to
* the sound card's current rate; scsynth default)
*/
def sampleRate: Int = sampleRateVar
/** The default sample rate is `0` (meaning that it is adjusted to
* the sound card's current rate; scsynth default)
*/
def sampleRate_=(value: Int): Unit = {
require (value >= 0)
sampleRateVar = value
}
private[this] var audioBuffersVar = 1024
/** The default number of audio buffers is `1024` (scsynth default).
* Must be greater than zero and a power of two.
*/
def audioBuffers: Int = audioBuffersVar
/** The default number of audio buffers is `1024` (scsynth default).
* Must be greater than zero and a power of two.
*/
def audioBuffers_=(value: Int): Unit = {
require (value > 0 && ri.isPowerOfTwo(value))
audioBuffersVar = value
}
/** The default maximum number of nodes is `1024` (scsynth default) */
var maxNodes: Int = 1024
/** The default maximum number of synth defs is `1024` (scsynth default) */
var maxSynthDefs: Int = 1024
/** The default memory size is `65536` (64 KB) (higher than scsynth's default of 8 KB) */
var memorySize: Int = 65536 // 8192
/** The default number of wire buffers is `256` (higher than scsynth's default of `64`). */
var wireBuffers: Int = 256 // 64
/** The default number of random number generators is `64` (scsynth default) */
var randomSeeds: Int = 64
/** The default setting for loading synth defs is `false` (this is not the scsynth default!) */
var loadSynthDefs: Boolean = false
/** The default settings for mach port name is `None` (scsynth default) */
var machPortName: Option[(String, String)] = None
/** The default verbosity level is `0` (scsynth default).
*
* '''Note:''' currently, decreasing the verbosity prevents the server connection
* to notice when the server has booted (issue no. 98).
*/
var verbosity: Int = 0
/** The default setting for plugin path redirection is `Nil`
* (use standard paths; scsynth default)
*/
var plugInsPaths: List[String] = Nil
/** The default setting for restricting file access is `None` (scsynth default) */
var restrictedPath: Option[String] = None
// ---- realtime only ----
/** (Realtime) The default host name is `127.0.0.1`. When booting, this is used
* to force scsynth to bind to a particular address (`-B` switch). To avoid the `-B`
* switch, you can use `"0.0.0.0"` (server will be reachable via network).
*/
var host: String = "127.0.0.1"
/** (Realtime) The default port is `57110`. */
var port: Int = 57110
/** (Realtime) The default transport is `UDP`. */
var transport: osc.Transport = UDP
/** (Realtime) The default settings for enabled input streams is `None` */
var inputStreamsEnabled: Option[String] = None
/** (Realtime) The default settings for enabled output streams is `None` */
var outputStreamsEnabled: Option[String] = None
private[this] var deviceNameVar = Option.empty[String]
private[this] var deviceNamesVar = Option.empty[(String, String)]
/** (Realtime) The default input/output device names is `None` (scsynth default; it will
* use the system default sound card)
*/
def deviceName: Option[String] = deviceNameVar
/** (Realtime) The default input/output device names is `None` (scsynth default; it will
* use the system default sound card)
*/
def deviceName_=(value: Option[String]): Unit = {
deviceNameVar = value
if (value.isDefined) deviceNamesVar = None
}
/** (Realtime) The default input/output device names is `None` (scsynth default; it will
* use the system default sound card)
*/
def deviceNames: Option[(String, String)] = deviceNamesVar
/** (Realtime) The default input/output device names is `None` (scsynth default; it will
* use the system default sound card)
*/
def deviceNames_=(value: Option[(String, String)]): Unit = {
deviceNamesVar = value
if (value.isDefined) deviceNameVar = None
}
private[this] var inputBusChannelsVar = 8
/** (Realtime) The default number of input bus channels is `8` (scsynth default) */
def inputBusChannels: Int = inputBusChannelsVar
/** (Realtime) The default number of input bus channels is `8` (scsynth default) */
def inputBusChannels_=(value: Int): Unit = {
require (value >= 0)
inputBusChannelsVar = value
}
/** (Realtime) The default setting for hardware block size is `0` (meaning that
* scsynth uses the hardware's current block size; scsynth default)
*/
var hardwareBlockSize: Int = 0
/** (Realtime) The default setting for zero-conf is `false` (other than
* scsynth's default which is `true`)
*/
var zeroConf: Boolean = false
/** (Realtime) The maximum number of TCP clients is `64` (scsynth default) */
var maxLogins: Int = 64
/** (Realtime) The default TCP session password is `None` */
var sessionPassword: Option[String] = None
// ---- non-realtime only ----
var nrtCommandPath : String = ""
var nrtInputPath : Option[String] = None
var nrtOutputPath : String = ""
var nrtHeaderFormat : AudioFileType = AudioFileType.AIFF
var nrtSampleFormat : SampleFormat = SampleFormat.Float
/** Picks and assigns a random free port for the server. This implies that
* the server will be running on the local machine.
*
* As a result, this method will change this config builder's `port` value.
* The caller must ensure that the `host` and `transport` fields have been
* decided on before calling this method. Later changes of either of these
* will render the result invalid.
*
* This method will fail with runtime exception if the host is not local.
*/
def pickPort(): Unit = {
require(isLocal)
transport match {
case UDP =>
val tmp = new DatagramSocket()
port = tmp.getLocalPort
tmp.close()
case TCP =>
val tmp = new ServerSocket(0)
port = tmp.getLocalPort
tmp.close()
case Browser =>
port = 57120 // default for now -- we could look into BrowserDriver free keys
}
}
/** Checks if the currently set `host` is located on the local machine. */
def isLocal: Boolean = {
val hostAddr = InetAddress.getByName(host)
hostAddr.isLoopbackAddress || hostAddr.isSiteLocalAddress || hostAddr.isAnyLocalAddress
}
def build: Config = {
val minAudioBuses = inputBusChannels + outputBusChannels + 1
val audioBusesAdjust = if (audioBusChannels >= minAudioBuses) audioBusChannels else {
ri.nextPowerOfTwo(minAudioBuses)
}
new Config(
program = program,
controlBusChannels = controlBusChannels,
audioBusChannels = audioBusesAdjust, /*audioBusChannels,*/
outputBusChannels = outputBusChannels,
blockSize = blockSize,
sampleRate = sampleRate,
audioBuffers = audioBuffers,
maxNodes = maxNodes,
maxSynthDefs = maxSynthDefs,
memorySize = memorySize,
wireBuffers = wireBuffers,
randomSeeds = randomSeeds,
loadSynthDefs = loadSynthDefs,
machPortName = machPortName,
verbosity = verbosity,
plugInsPaths = plugInsPaths,
restrictedPath = restrictedPath,
/* memoryLocking, */
host = host,
port = port,
transport = transport,
inputStreamsEnabled = inputStreamsEnabled,
outputStreamsEnabled = outputStreamsEnabled,
deviceNames = deviceNames,
deviceName = deviceName,
inputBusChannels = inputBusChannels,
hardwareBlockSize = hardwareBlockSize,
zeroConf = zeroConf,
maxLogins = maxLogins,
sessionPassword = sessionPassword,
nrtCommandPath = nrtCommandPath,
nrtInputPath = nrtInputPath,
nrtOutputPath = nrtOutputPath,
nrtHeaderFormat = nrtHeaderFormat,
nrtSampleFormat = nrtSampleFormat
)
}
def read(config: Config): Unit = {
program = config.program
controlBusChannels = config.controlBusChannels
audioBusChannels = config.audioBusChannels
outputBusChannels = config.outputBusChannels
blockSize = config.blockSize
sampleRate = config.sampleRate
audioBuffers = config.audioBuffers
maxNodes = config.maxNodes
maxSynthDefs = config.maxSynthDefs
memorySize = config.memorySize
wireBuffers = config.wireBuffers
randomSeeds = config.randomSeeds
loadSynthDefs = config.loadSynthDefs
machPortName = config.machPortName
verbosity = config.verbosity
plugInsPaths = config.plugInsPaths
restrictedPath = config.restrictedPath
host = config.host
port = config.port
transport = config.transport
inputStreamsEnabled = config.inputStreamsEnabled
outputStreamsEnabled= config.outputStreamsEnabled
deviceNames = config.deviceNames
deviceName = config.deviceName
inputBusChannels = config.inputBusChannels
hardwareBlockSize = config.hardwareBlockSize
zeroConf = config.zeroConf
maxLogins = config.maxLogins
sessionPassword = config.sessionPassword
nrtCommandPath = config.nrtCommandPath
nrtInputPath = config.nrtInputPath
nrtOutputPath = config.nrtOutputPath
nrtHeaderFormat = config.nrtHeaderFormat
nrtSampleFormat = config.nrtSampleFormat
}
}
def boot: ServerConnection = boot()()
def boot(name: String = "localhost", config: Config = Config().build,
clientConfig: Client.Config = Client.Config().build)
(listener: ServerConnection.Listener = PartialFunction.empty): ServerConnection = {
val sc = initBoot(name, config, clientConfig)
if (!(listener eq PartialFunction.empty)) sc.addListener(listener)
sc.start()
sc
}
private def initBoot(name: String = "localhost", config: Config,
clientConfig: Client.Config = Client.Config().build) = {
val (addr, c) = prepareConnection(config, clientConfig)
new impl.Booting(name, c, addr, config, clientConfig, true)
}
def connect: ServerConnection = connect()()
def connect(name: String = "localhost", config: Config = Config().build,
clientConfig: Client.Config = Client.Config().build)
(listener: ServerConnection.Listener = PartialFunction.empty): ServerConnection = {
val (addr, c) = prepareConnection(config, clientConfig)
val sc = new impl.Connection(name, c, addr, config, clientConfig, true)
if (!(listener eq PartialFunction.empty)) sc.addListener(listener)
sc.start()
sc
}
def run(code: Server => Unit): Unit = run()(code)
/** Utility method to test code quickly with a running server. This boots a
* server and executes the passed in code when the server is up. A shutdown
* hook is registered to make sure the server is destroyed when the VM exits.
*/
def run(config: Config = Config().build)(code: Server => Unit): Unit = {
// val b = boot( config = config )
val sync = new AnyRef
var s: Server = null
val sc = initBoot(config = config)
val li: ServerConnection.Listener = {
case ServerConnection.Running(srv) => sync.synchronized {
s = srv
}; code(srv)
}
sc.addListener(li)
Runtime.getRuntime.addShutdownHook(new Thread {
override def run(): Unit =
sync.synchronized {
if (s != null) {
if (s.condition != Server.Offline) s.quit()
} else sc.abort()
}
})
sc.start()
}
/** Creates an unconnected server proxy. This may be useful for creating NRT command files.
* Any attempt to try to send messages to the server will fail.
*/
def dummy(name: String = "dummy", config: Config = Config().build,
clientConfig: Client.Config = Client.Config().build): Server = {
val sr = config.sampleRate
val status = message.StatusReply(numUGens = 0, numSynths = 0, numGroups = 0, numDefs = 0,
avgCPU = 0f, peakCPU = 0f, sampleRate = sr, actualSampleRate = sr)
new impl.OfflineServerImpl(name, /* c, addr, */ config, clientConfig, status)
}
def allocPort(transport: osc.Transport): Int = {
transport match {
case TCP =>
val ss = new ServerSocket(0)
try {
ss.getLocalPort
} finally {
ss.close()
}
case UDP =>
val ds = new DatagramSocket()
try {
ds.getLocalPort
} finally {
ds.close()
}
case other => sys.error(s"Unsupported transport : ${other.name}")
}
}
def printError(name: String, t: Throwable): Unit = {
println(s"$name :")
t.printStackTrace()
}
implicit def defaultGroup(s: Server): Group = s.defaultGroup
type Listener = Model.Listener[Update]
sealed trait Update
sealed trait Condition extends Update
case object Running extends Condition
case object Offline extends Condition
private[synth] case object NoPending extends Condition
final case class Counts(c: message.StatusReply) extends Update
/** Starts an NRT rendering process based on the NRT parameters of the configuration argument.
*
* '''Note:''' The returned process must be explicitly started by calling `start()`
*
* @param dur the duration of the bounce, used to emit process updates
* @param config the server configuration in which `nrtCommandPath` must be set
*
* @return the process whose return value is the process exit code of scsynth (0 indicating success)
*/
def renderNRT(dur: Double, config: Server.Config): Processor[Int] with Processor.Prepared =
new impl.NRTImpl(dur, config)
def version: Try[(String, String)] = version()
def version(config: Config = Config().build): Try[(String, String)] = Try {
import scala.sys.process._
val output = Seq(config.program, "-v").!!
val i = output.indexOf(' ') + 1
val j0 = output.indexOf(' ', i)
val j1 = if (j0 > i) j0 else output.indexOf('\n', i)
val j = if (j1 > i) j1 else output.length
val k = output.indexOf('(', j) + 1
val m = output.indexOf(')', k)
val version = output.substring(i, j)
val build = if (m > k) output.substring(k, m) else ""
(version, build)
}
}
sealed trait ServerLike {
def name : String
def config: Server.Config
def addr : Server.Address // InetSocketAddress
}
object ServerConnection {
type Listener = Model.Listener[Condition]
sealed abstract class Condition
case class Preparing(server: Server) extends Condition
case class Running (server: Server) extends Condition
case object Aborted extends Condition
}
trait ServerConnection extends ServerLike with Model[ServerConnection.Condition] {
def abort(): Unit
}
/** The client-side representation of the SuperCollider server.
*
* Additional operations are available by importing `Ops._`.
*/
trait Server extends ServerLike with Model[Server.Update] {
server =>
import de.sciss.synth.Server._
val clientConfig : Client.Config
def rootNode : Group
def defaultGroup : Group
def nodeManager : NodeManager
def bufManager : BufferManager
def isLocal : Boolean
def isConnected : Boolean
def isRunning : Boolean
def isOffline : Boolean
def nextNodeId(): Int
def nextSyncId(): Int
def allocControlBus(numChannels: Int): Int
def allocAudioBus (numChannels: Int): Int
def freeControlBus(index: Int): Unit
def freeAudioBus (index: Int): Unit
def allocBuffer(numChannels: Int): Int
def freeBuffer (index : Int): Unit
/** Sends out an OSC packet without waiting for any replies. */
def ! (p: osc.Packet): Unit
/** Sends out an OSC packet that generates some kind of reply, and
* returns immediately. It registers a handler to parse that reply.
* The handler is tested for each incoming OSC message (using its
* `isDefinedAt` method) and invoked and removed in case of a
* match, completing the returned future.
*
* If the handler does not match in the given timeout period,
* the future fails with a `Timeout` exception, and the handler is removed.
*
* @param packet the packet to send out
* @param timeout the timeout duration
* @param handler the handler to match against incoming messages
* @return a future of the successfully completed handler or timeout exception
*
* @see [[de.sciss.synth.message.Timeout]]
*/
def !! [A](packet: osc.Packet, timeout: Duration = 6.seconds)(handler: PartialFunction[osc.Message, A]): Future[A]
/** The last reported server data, such as number of synths and groups, sample rate. */
def counts: message.StatusReply
/** Shortcut to `counts.sampleRate`. */
def sampleRate: Double
def condition: Condition
/** Starts a repeatedly running status watcher that updates the `condition` and `counts`
* information.
*/
def startAliveThread(delay: Float = 0.25f, period: Float = 0.25f, deathBounces: Int = 25): Unit
def stopAliveThread(): Unit
/** Shortcut to `this ! message.Status`. If the 'alive thread' is running,
* it will take care of querying the counts frequently.
*/
def queryCounts(): Unit
/** Allocates a new unique synchronization identifier,
* and returns the corresponding `/sync` message.
*/
def syncMsg(): message.Sync
def dumpOSC (mode: osc.Dump = osc.Dump.Text, filter: osc.Packet => Boolean = _ => true): Unit
def dumpInOSC (mode: osc.Dump = osc.Dump.Text, filter: osc.Packet => Boolean = _ => true): Unit
def dumpOutOSC(mode: osc.Dump = osc.Dump.Text, filter: osc.Packet => Boolean = _ => true): Unit
/** Sends a `quitMsg` and then invokes `dispose()`. */
def quit(): Unit
def quitMsg: message.ServerQuit.type
/** Disconnects the client, and frees any resources on the client-side. */
def dispose(): Unit
private[synth] def addResponder (resp: message.Responder): Unit
private[synth] def removeResponder(resp: message.Responder): Unit
override def toString = s"<$name>"
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy