Skip to content

Commit 6a77567

Browse files
Merge branch 'sean/tool-annotations' of https://github.com/SeanChinJunKai/kotlin-sdk into sean/tool-annotations
2 parents 84f12f4 + 0889d48 commit 6a77567

16 files changed

+188
-101
lines changed

.idea/misc.xml

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

build.gradle.kts

-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,6 @@ plugins {
1111
alias(libs.plugins.kotlin.serialization)
1212
alias(libs.plugins.dokka)
1313
alias(libs.plugins.jreleaser)
14-
alias(libs.plugins.atomicfu)
1514
`maven-publish`
1615
alias(libs.plugins.kotlinx.binary.compatibility.validator)
1716
}

gradle.properties

+4
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,8 @@
11
kotlin.code.style=official
2+
org.gradle.jvmargs=-XX:+UseParallelGC
3+
org.gradle.parallel=true
4+
org.gradle.configuration-cache=true
5+
org.gradle.configuration-cache.parallel=true
26

37
org.jetbrains.dokka.experimental.gradle.pluginMode=V2Enabled
48
org.jetbrains.dokka.experimental.gradle.pluginMode.noWarn=true

gradle/libs.versions.toml

+1-3
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,7 @@
11
[versions]
22
# plugins version
3-
kotlin = "2.0.21"
3+
kotlin = "2.1.20"
44
dokka = "2.0.0"
5-
atomicfu = "0.26.1"
65

76
# libraries version
87
serialization = "1.7.3"
@@ -40,5 +39,4 @@ kotlin-multiplatform = { id = "org.jetbrains.kotlin.multiplatform", version.ref
4039
kotlin-serialization = { id = "org.jetbrains.kotlin.plugin.serialization", version.ref = "kotlin" }
4140
dokka = { id = "org.jetbrains.dokka", version.ref = "dokka" }
4241
jreleaser = { id = "org.jreleaser", version.ref = "jreleaser"}
43-
atomicfu = { id = "org.jetbrains.kotlinx.atomicfu", version.ref = "atomicfu" }
4442
kotlinx-binary-compatibility-validator = { id = "org.jetbrains.kotlinx.binary-compatibility-validator", version.ref = "binaryCompatibilityValidatorPlugin" }

gradle/wrapper/gradle-wrapper.jar

-16.7 KB
Binary file not shown.
+3-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
1-
#Thu Dec 12 15:35:46 CET 2024
21
distributionBase=GRADLE_USER_HOME
32
distributionPath=wrapper/dists
4-
distributionUrl=https\://services.gradle.org/distributions/gradle-8.10-bin.zip
3+
distributionUrl=https\://services.gradle.org/distributions/gradle-8.13-bin.zip
4+
networkTimeout=10000
5+
validateDistributionUrl=true
56
zipStoreBase=GRADLE_USER_HOME
67
zipStorePath=wrapper/dists

gradlew

+30-13
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,8 @@
1515
# See the License for the specific language governing permissions and
1616
# limitations under the License.
1717
#
18+
# SPDX-License-Identifier: Apache-2.0
19+
#
1820

1921
##############################################################################
2022
#
@@ -55,7 +57,7 @@
5557
# Darwin, MinGW, and NonStop.
5658
#
5759
# (3) This script is generated from the Groovy template
58-
# https://github.com/gradle/gradle/blob/master/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
60+
# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
5961
# within the Gradle project.
6062
#
6163
# You can find Gradle at https://github.com/gradle/gradle/.
@@ -80,13 +82,11 @@ do
8082
esac
8183
done
8284

83-
APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit
84-
85-
APP_NAME="Gradle"
85+
# This is normally unused
86+
# shellcheck disable=SC2034
8687
APP_BASE_NAME=${0##*/}
87-
88-
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
89-
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
88+
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
89+
APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s\n' "$PWD" ) || exit
9090

9191
# Use the maximum available, or set MAX_FD != -1 to use that value.
9292
MAX_FD=maximum
@@ -133,22 +133,29 @@ location of your Java installation."
133133
fi
134134
else
135135
JAVACMD=java
136-
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
136+
if ! command -v java >/dev/null 2>&1
137+
then
138+
die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
137139
138140
Please set the JAVA_HOME variable in your environment to match the
139141
location of your Java installation."
142+
fi
140143
fi
141144

142145
# Increase the maximum file descriptors if we can.
143146
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
144147
case $MAX_FD in #(
145148
max*)
149+
# In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
150+
# shellcheck disable=SC2039,SC3045
146151
MAX_FD=$( ulimit -H -n ) ||
147152
warn "Could not query maximum file descriptor limit"
148153
esac
149154
case $MAX_FD in #(
150155
'' | soft) :;; #(
151156
*)
157+
# In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
158+
# shellcheck disable=SC2039,SC3045
152159
ulimit -n "$MAX_FD" ||
153160
warn "Could not set maximum file descriptor limit to $MAX_FD"
154161
esac
@@ -193,18 +200,28 @@ if "$cygwin" || "$msys" ; then
193200
done
194201
fi
195202

196-
# Collect all arguments for the java command;
197-
# * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of
198-
# shell script including quotes and variable substitutions, so put them in
199-
# double quotes to make sure that they get re-expanded; and
200-
# * put everything else in single quotes, so that it's not re-expanded.
203+
204+
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
205+
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
206+
207+
# Collect all arguments for the java command:
208+
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
209+
# and any embedded shellness will be escaped.
210+
# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
211+
# treated as '${Hostname}' itself on the command line.
201212

202213
set -- \
203214
"-Dorg.gradle.appname=$APP_BASE_NAME" \
204215
-classpath "$CLASSPATH" \
205216
org.gradle.wrapper.GradleWrapperMain \
206217
"$@"
207218

219+
# Stop when "xargs" is not available.
220+
if ! command -v xargs >/dev/null 2>&1
221+
then
222+
die "xargs is not available"
223+
fi
224+
208225
# Use "xargs" to parse quoted args.
209226
#
210227
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.

gradlew.bat

+21-16
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,10 @@
1313
@rem See the License for the specific language governing permissions and
1414
@rem limitations under the License.
1515
@rem
16+
@rem SPDX-License-Identifier: Apache-2.0
17+
@rem
1618

17-
@if "%DEBUG%" == "" @echo off
19+
@if "%DEBUG%"=="" @echo off
1820
@rem ##########################################################################
1921
@rem
2022
@rem Gradle startup script for Windows
@@ -25,7 +27,8 @@
2527
if "%OS%"=="Windows_NT" setlocal
2628

2729
set DIRNAME=%~dp0
28-
if "%DIRNAME%" == "" set DIRNAME=.
30+
if "%DIRNAME%"=="" set DIRNAME=.
31+
@rem This is normally unused
2932
set APP_BASE_NAME=%~n0
3033
set APP_HOME=%DIRNAME%
3134

@@ -40,13 +43,13 @@ if defined JAVA_HOME goto findJavaFromJavaHome
4043

4144
set JAVA_EXE=java.exe
4245
%JAVA_EXE% -version >NUL 2>&1
43-
if "%ERRORLEVEL%" == "0" goto execute
46+
if %ERRORLEVEL% equ 0 goto execute
4447

45-
echo.
46-
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
47-
echo.
48-
echo Please set the JAVA_HOME variable in your environment to match the
49-
echo location of your Java installation.
48+
echo. 1>&2
49+
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2
50+
echo. 1>&2
51+
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
52+
echo location of your Java installation. 1>&2
5053

5154
goto fail
5255

@@ -56,11 +59,11 @@ set JAVA_EXE=%JAVA_HOME%/bin/java.exe
5659

5760
if exist "%JAVA_EXE%" goto execute
5861

59-
echo.
60-
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
61-
echo.
62-
echo Please set the JAVA_HOME variable in your environment to match the
63-
echo location of your Java installation.
62+
echo. 1>&2
63+
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2
64+
echo. 1>&2
65+
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
66+
echo location of your Java installation. 1>&2
6467

6568
goto fail
6669

@@ -75,13 +78,15 @@ set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
7578

7679
:end
7780
@rem End local scope for the variables with windows NT shell
78-
if "%ERRORLEVEL%"=="0" goto mainEnd
81+
if %ERRORLEVEL% equ 0 goto mainEnd
7982

8083
:fail
8184
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
8285
rem the _cmd.exe /c_ return code!
83-
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
84-
exit /b 1
86+
set EXIT_CODE=%ERRORLEVEL%
87+
if %EXIT_CODE% equ 0 set EXIT_CODE=1
88+
if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
89+
exit /b %EXIT_CODE%
8590

8691
:mainEnd
8792
if "%OS%"=="Windows_NT" endlocal

src/commonMain/kotlin/io/modelcontextprotocol/kotlin/sdk/client/Client.kt

+6
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import kotlinx.serialization.json.JsonElement
1010
import kotlinx.serialization.json.JsonNull
1111
import kotlinx.serialization.json.JsonObject
1212
import kotlinx.serialization.json.JsonPrimitive
13+
import kotlin.coroutines.cancellation.CancellationException
1314

1415
/**
1516
* Options for configuring the MCP client.
@@ -100,7 +101,12 @@ public open class Client(
100101
notification(InitializedNotification())
101102
} catch (error: Throwable) {
102103
close()
104+
if (error !is CancellationException) {
105+
throw IllegalStateException("Error connecting to transport: ${error.message}")
106+
}
107+
103108
throw error
109+
104110
}
105111
}
106112

src/commonMain/kotlin/io/modelcontextprotocol/kotlin/sdk/client/SSEClientTransport.kt

+5-4
Original file line numberDiff line numberDiff line change
@@ -8,10 +8,10 @@ import io.ktor.http.*
88
import io.modelcontextprotocol.kotlin.sdk.JSONRPCMessage
99
import io.modelcontextprotocol.kotlin.sdk.shared.AbstractTransport
1010
import io.modelcontextprotocol.kotlin.sdk.shared.McpJson
11-
import kotlinx.atomicfu.AtomicBoolean
12-
import kotlinx.atomicfu.atomic
1311
import kotlinx.coroutines.*
1412
import kotlinx.serialization.encodeToString
13+
import kotlin.concurrent.atomics.AtomicBoolean
14+
import kotlin.concurrent.atomics.ExperimentalAtomicApi
1515
import kotlin.properties.Delegates
1616
import kotlin.time.Duration
1717

@@ -22,6 +22,7 @@ public typealias SSEClientTransport = SseClientTransport
2222
* Client transport for SSE: this will connect to a server using Server-Sent Events for receiving
2323
* messages and make separate POST requests for sending messages.
2424
*/
25+
@OptIn(ExperimentalAtomicApi::class)
2526
public class SseClientTransport(
2627
private val client: HttpClient,
2728
private val urlString: String?,
@@ -32,7 +33,7 @@ public class SseClientTransport(
3233
CoroutineScope(session.coroutineContext + SupervisorJob())
3334
}
3435

35-
private val initialized: AtomicBoolean = atomic(false)
36+
private val initialized: AtomicBoolean = AtomicBoolean(false)
3637
private var session: ClientSSESession by Delegates.notNull()
3738
private val endpoint = CompletableDeferred<String>()
3839

@@ -127,7 +128,7 @@ public class SseClientTransport(
127128
}
128129

129130
override suspend fun close() {
130-
if (!initialized.value) {
131+
if (!initialized.load()) {
131132
error("SSEClientTransport is not initialized!")
132133
}
133134

src/commonMain/kotlin/io/modelcontextprotocol/kotlin/sdk/client/StdioClientTransport.kt

+6-10
Original file line numberDiff line numberDiff line change
@@ -5,17 +5,12 @@ import io.modelcontextprotocol.kotlin.sdk.JSONRPCMessage
55
import io.modelcontextprotocol.kotlin.sdk.shared.AbstractTransport
66
import io.modelcontextprotocol.kotlin.sdk.shared.ReadBuffer
77
import io.modelcontextprotocol.kotlin.sdk.shared.serializeMessage
8-
import kotlinx.atomicfu.AtomicBoolean
9-
import kotlinx.atomicfu.atomic
108
import kotlinx.coroutines.*
119
import kotlinx.coroutines.channels.Channel
1210
import kotlinx.coroutines.channels.consumeEach
13-
import kotlinx.io.Buffer
14-
import kotlinx.io.Sink
15-
import kotlinx.io.Source
16-
import kotlinx.io.buffered
17-
import kotlinx.io.readByteArray
18-
import kotlinx.io.writeString
11+
import kotlinx.io.*
12+
import kotlin.concurrent.atomics.AtomicBoolean
13+
import kotlin.concurrent.atomics.ExperimentalAtomicApi
1914
import kotlin.coroutines.CoroutineContext
2015

2116
/**
@@ -27,6 +22,7 @@ import kotlin.coroutines.CoroutineContext
2722
* @param input The input stream where messages are received.
2823
* @param output The output stream where messages are sent.
2924
*/
25+
@OptIn(ExperimentalAtomicApi::class)
3026
public class StdioClientTransport(
3127
private val input: Source,
3228
private val output: Sink
@@ -37,7 +33,7 @@ public class StdioClientTransport(
3733
CoroutineScope(ioCoroutineContext + SupervisorJob())
3834
}
3935
private var job: Job? = null
40-
private val initialized: AtomicBoolean = atomic(false)
36+
private val initialized: AtomicBoolean = AtomicBoolean(false)
4137
private val sendChannel = Channel<JSONRPCMessage>(Channel.UNLIMITED)
4238
private val readBuffer = ReadBuffer()
4339

@@ -96,7 +92,7 @@ public class StdioClientTransport(
9692
}
9793

9894
override suspend fun send(message: JSONRPCMessage) {
99-
if (!initialized.value) {
95+
if (!initialized.load()) {
10096
error("Transport not started")
10197
}
10298

src/commonMain/kotlin/io/modelcontextprotocol/kotlin/sdk/server/SSEServerTransport.kt

+6-5
Original file line numberDiff line numberDiff line change
@@ -8,10 +8,10 @@ import io.ktor.server.sse.*
88
import io.modelcontextprotocol.kotlin.sdk.JSONRPCMessage
99
import io.modelcontextprotocol.kotlin.sdk.shared.AbstractTransport
1010
import io.modelcontextprotocol.kotlin.sdk.shared.McpJson
11-
import kotlinx.atomicfu.AtomicBoolean
12-
import kotlinx.atomicfu.atomic
1311
import kotlinx.coroutines.job
1412
import kotlinx.serialization.encodeToString
13+
import kotlin.concurrent.atomics.AtomicBoolean
14+
import kotlin.concurrent.atomics.ExperimentalAtomicApi
1515
import kotlin.uuid.ExperimentalUuidApi
1616
import kotlin.uuid.Uuid
1717

@@ -25,11 +25,12 @@ public typealias SSEServerTransport = SseServerTransport
2525
*
2626
* Creates a new SSE server transport, which will direct the client to POST messages to the relative or absolute URL identified by `_endpoint`.
2727
*/
28+
@OptIn(ExperimentalAtomicApi::class)
2829
public class SseServerTransport(
2930
private val endpoint: String,
3031
private val session: ServerSSESession,
3132
) : AbstractTransport() {
32-
private val initialized: AtomicBoolean = atomic(false)
33+
private val initialized: AtomicBoolean = AtomicBoolean(false)
3334

3435
@OptIn(ExperimentalUuidApi::class)
3536
public val sessionId: String = Uuid.random().toString()
@@ -63,7 +64,7 @@ public class SseServerTransport(
6364
* This should be called when a POST request is made to send a message to the server.
6465
*/
6566
public suspend fun handlePostMessage(call: ApplicationCall) {
66-
if (!initialized.value) {
67+
if (!initialized.load()) {
6768
val message = "SSE connection not established"
6869
call.respondText(message, status = HttpStatusCode.InternalServerError)
6970
_onError.invoke(IllegalStateException(message))
@@ -112,7 +113,7 @@ public class SseServerTransport(
112113
}
113114

114115
override suspend fun send(message: JSONRPCMessage) {
115-
if (!initialized.value) {
116+
if (!initialized.load()) {
116117
throw error("Not connected")
117118
}
118119

0 commit comments

Comments
 (0)