如何在Supercollider和Python应用之间调试OSC协议

时间:2018-10-28 13:29:14

标签: python osc supercollider

是SuperCollider的新手,我正在使用一个教程来尝试让Pycharm应用程序与Supercollider对话(Python版本:3.7 / SuperCollider 3.9.3)。在客户端,我尝试使用pythonosc和OSC:

pythosc代码:

import argparse
import random
from pythonosc import osc_message_builder
from pythonosc import udp_client
import socket

if __name__ == "__main__":

    parser = argparse.ArgumentParser()
    parser.add_argument("--ip", default='127.0.0.1',
                        help="The ip of the OSC server")
    parser.add_argument("--port", type=int, default=57110,
                        help="The port the OSC server is listening on")
    args = parser.parse_args()

    client = udp_client.SimpleUDPClient(args.ip, args.port)

    client.send_message("/print", 500)

OSC代码:

import OSC
import time, random

if __name__ == "__main__":
    client = OSC.OSCClient()
    client.connect(("127.0.0.1", 57110))
    msg = OSC.OSCMessage()
    msg.setAddress("/print")
    msg.append(500)
    client.send(msg)

SuperCollider中的代码:

s.boot;

(
SynthDef( \sin, { | amp = 0.01, freq = 333, trig = 1 |
    var env, sig;
    env = EnvGen.kr( Env.asr( 0.001, 0.9, 0.001 ), trig, doneAction: 0 );
    sig = LFTri.ar( [ freq, freq * 0.999 ], 0.0, amp ) * env;
    Out.ar( [ 0 ], sig * 0.6 );
}).add;

h = Synth( \sin, [ \amp, 0.4 ] );

x = OSCFunc( { | msg, time, addr, port |
    var pyFreq;

    pyFreq = msg[1].asFloat;
    ( "freq is " + pyFreq ).postln;
    h.set( \freq, pyFreq );
}, '/print' );
)

同时使用pythonosc和OSC代码时,我得到:“服务器失败:/ print命令未找到” 。显然,python应用正在与SC建立通信,但无法解释'/ print'。我注意到SC服务器在端口57110上启动,但NetAddr.langPort返回为57120。我不确定它们为什么不同。我整个晚上都在网上审查可能的解决方案,但没有运气。其他两个StackOverflow用户也发布了类似的问题,其中一个暗示可能是端口问题。在目前的情况下,我不知道它是怎么回事,在这种情况下,SC根本什么也不会。 (用python代码将端口更改为57120导致SC服务器无响应)。从人们以前对这种柏忌的经历中得出的任何建议都将受到欢迎,因为从根本上说,它正在驱使我。非常慢,坚果!

1 个答案:

答案 0 :(得分:0)

简短答案:使用NetAddr.langPort返回的端口。

SuperCollider服务器(scsynth)和语言进程(sclang)是不同的东西。如果您使用SuperCollider IDE,您可能会感到困惑,因为这两个过程的输出都发布在该窗口的发布窗口中,并且初学者也很难理解这一区别。

您非常接近答案-您注意到服务器的端口与语言的端口不同。为了使两个进程通过基于UDP的OSC相互通信,它们需要使用单独的端口。服务器使用OSC地址来实现命令协议,您可以在其中找到here的文档。如您所见,有一些命令用于创建新的合成节点(/s_new),分配缓冲区(/b_alloc)和在控制总线上设置值(/s_set),但没有/print的命令。这就是为什么您看到该特定错误消息的原因:“服务器失败:/ print命令未找到” 。 (顺便说一下,该消息来自服务器,而不是语言。)

另一方面,

OSCFunc对象响应进入语言过程的OSC消息。因此,您需要做的就是将OSC消息从Python发送到正确的端口-57120。

This documentation article可能有助于理解SuperCollider服务器(scsynth)和客户端(Python,sclang或您发送OSC消息的任何其他程序)之间的区别。