mirror of
				https://github.com/HappyTanuki/BumbleCee.git
				synced 2025-10-26 09:55:14 +00:00 
			
		
		
		
	Compare commits
	
		
			19 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 0cb7ea8225 | |||
| 7e43900f22 | |||
| c2cb16c6d7 | |||
| 3186561818 | |||
| 30569f4472 | |||
| b4476c68d7 | |||
| c276065a65 | |||
| 68b6105ac3 | |||
| 0633855374 | |||
| b9ab9ac60f | |||
| 8607c4a8a5 | |||
| 51f9b25bd8 | |||
| 3f1edbbf16 | |||
| ba56fe015f | |||
| 5c99d42fe3 | |||
| e0e4b17675 | |||
| e62a862e42 | |||
| 75a34d12b6 | |||
| 4e4856e961 | 
							
								
								
									
										13
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,6 @@ | |||||||
| build/ | build/* | ||||||
| out/ | Temp/ | ||||||
| .vs/ | Music | ||||||
| .vscode/ | *.json | ||||||
| .idea/ | yt-dlp | ||||||
| tmp/ | ffmpeg | ||||||
| config.json |  | ||||||
							
								
								
									
										10
									
								
								.idea/.gitignore
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								.idea/.gitignore
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | # Default ignored files | ||||||
|  | /shelf/ | ||||||
|  | /workspace.xml | ||||||
|  | # Editor-based HTTP Client requests | ||||||
|  | /httpRequests/ | ||||||
|  | # Environment-dependent path to Maven home directory | ||||||
|  | /mavenHomeManager.xml | ||||||
|  | # Datasource local storage ignored files | ||||||
|  | /dataSources/ | ||||||
|  | /dataSources.local.xml | ||||||
							
								
								
									
										9
									
								
								.idea/BumbleCee.iml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								.idea/BumbleCee.iml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <module type="JAVA_MODULE" version="4"> | ||||||
|  |   <component name="NewModuleRootManager" inherit-compiler-output="true"> | ||||||
|  |     <exclude-output /> | ||||||
|  |     <content url="file://$MODULE_DIR$" /> | ||||||
|  |     <orderEntry type="inheritedJdk" /> | ||||||
|  |     <orderEntry type="sourceFolder" forTests="false" /> | ||||||
|  |   </component> | ||||||
|  | </module> | ||||||
							
								
								
									
										6
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <project version="4"> | ||||||
|  |   <component name="ProjectRootManager" version="2" languageLevel="JDK_23" default="true" project-jdk-name="openjdk-23" project-jdk-type="JavaSDK"> | ||||||
|  |     <output url="file://$PROJECT_DIR$/out" /> | ||||||
|  |   </component> | ||||||
|  | </project> | ||||||
							
								
								
									
										8
									
								
								.idea/modules.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.idea/modules.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <project version="4"> | ||||||
|  |   <component name="ProjectModuleManager"> | ||||||
|  |     <modules> | ||||||
|  |       <module fileurl="file://$PROJECT_DIR$/.idea/BumbleCee.iml" filepath="$PROJECT_DIR$/.idea/BumbleCee.iml" /> | ||||||
|  |     </modules> | ||||||
|  |   </component> | ||||||
|  | </project> | ||||||
							
								
								
									
										6
									
								
								.idea/vcs.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.idea/vcs.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <project version="4"> | ||||||
|  |   <component name="VcsDirectoryMappings"> | ||||||
|  |     <mapping directory="" vcs="Git" /> | ||||||
|  |   </component> | ||||||
|  | </project> | ||||||
							
								
								
									
										22
									
								
								.vscode/c_cpp_properties.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								.vscode/c_cpp_properties.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | { | ||||||
|  |     "configurations": [ | ||||||
|  |         { | ||||||
|  |             "name": "Linux", | ||||||
|  |             "includePath": [ | ||||||
|  |                 "${workspaceFolder}/**", | ||||||
|  |                 "${workspaceFolder}/include" | ||||||
|  |             ], | ||||||
|  |             "defines": [ | ||||||
|  |                 "DDPP_CORO=on" | ||||||
|  |             ], | ||||||
|  |             "compilerPath": "/usr/bin/gcc", | ||||||
|  |             "cStandard": "c17", | ||||||
|  |             "cppStandard": "c++20", | ||||||
|  |             "intelliSenseMode": "linux-gcc-x64", | ||||||
|  |             "compilerArgs": [ | ||||||
|  |                 "-DDPP_CORO" | ||||||
|  |             ] | ||||||
|  |         } | ||||||
|  |     ], | ||||||
|  |     "version": 4 | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @@ -5,7 +5,7 @@ | |||||||
|             "name": "Debug", |             "name": "Debug", | ||||||
|             "type": "cppdbg", |             "type": "cppdbg", | ||||||
|             "request": "launch", |             "request": "launch", | ||||||
|             "program": "/home/happytanuki/BumbleCee/build/BumbleCee", |             "program": "${workspaceFolder}/build/BumbleCee", | ||||||
|             "stopAtEntry": false, |             "stopAtEntry": false, | ||||||
|             "cwd": "${workspaceFolder}", |             "cwd": "${workspaceFolder}", | ||||||
|             "environment": [], |             "environment": [], | ||||||
|   | |||||||
							
								
								
									
										147
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										147
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,78 +1,107 @@ | |||||||
| { | { | ||||||
|     "files.associations": { |     "files.associations": { | ||||||
|         "iostream": "cpp", |         "iosfwd": "cpp", | ||||||
|         "any": "cpp", |  | ||||||
|         "chrono": "cpp", |  | ||||||
|         "codecvt": "cpp", |  | ||||||
|         "compare": "cpp", |  | ||||||
|         "condition_variable": "cpp", |  | ||||||
|         "deque": "cpp", |  | ||||||
|         "forward_list": "cpp", |  | ||||||
|         "list": "cpp", |  | ||||||
|         "string": "cpp", |  | ||||||
|         "unordered_map": "cpp", |  | ||||||
|         "vector": "cpp", |  | ||||||
|         "exception": "cpp", |  | ||||||
|         "functional": "cpp", |  | ||||||
|         "iterator": "cpp", |  | ||||||
|         "memory": "cpp", |  | ||||||
|         "memory_resource": "cpp", |  | ||||||
|         "optional": "cpp", |  | ||||||
|         "string_view": "cpp", |  | ||||||
|         "random": "cpp", |  | ||||||
|         "*.tcc": "cpp", |  | ||||||
|         "fstream": "cpp", |  | ||||||
|         "future": "cpp", |  | ||||||
|         "initializer_list": "cpp", |  | ||||||
|         "istream": "cpp", |  | ||||||
|         "mutex": "cpp", |  | ||||||
|         "new": "cpp", |  | ||||||
|         "ostream": "cpp", |  | ||||||
|         "ranges": "cpp", |  | ||||||
|         "semaphore": "cpp", |  | ||||||
|         "shared_mutex": "cpp", |  | ||||||
|         "sstream": "cpp", |         "sstream": "cpp", | ||||||
|         "stdexcept": "cpp", |  | ||||||
|         "stop_token": "cpp", |  | ||||||
|         "streambuf": "cpp", |  | ||||||
|         "system_error": "cpp", |  | ||||||
|         "thread": "cpp", |  | ||||||
|         "tuple": "cpp", |  | ||||||
|         "type_traits": "cpp", |  | ||||||
|         "typeinfo": "cpp", |  | ||||||
|         "valarray": "cpp", |  | ||||||
|         "variant": "cpp", |  | ||||||
|         "array": "cpp", |  | ||||||
|         "atomic": "cpp", |  | ||||||
|         "bit": "cpp", |  | ||||||
|         "cctype": "cpp", |         "cctype": "cpp", | ||||||
|         "charconv": "cpp", |  | ||||||
|         "clocale": "cpp", |         "clocale": "cpp", | ||||||
|         "cmath": "cpp", |         "cmath": "cpp", | ||||||
|         "concepts": "cpp", |  | ||||||
|         "coroutine": "cpp", |  | ||||||
|         "csignal": "cpp", |         "csignal": "cpp", | ||||||
|         "cstdarg": "cpp", |         "cstdarg": "cpp", | ||||||
|         "cstddef": "cpp", |         "cstddef": "cpp", | ||||||
|         "cstdint": "cpp", |  | ||||||
|         "cstdio": "cpp", |         "cstdio": "cpp", | ||||||
|         "cstdlib": "cpp", |         "cstdlib": "cpp", | ||||||
|         "cstring": "cpp", |         "cstring": "cpp", | ||||||
|         "ctime": "cpp", |         "ctime": "cpp", | ||||||
|         "cwchar": "cpp", |         "cwchar": "cpp", | ||||||
|         "cwctype": "cpp", |         "cwctype": "cpp", | ||||||
|         "map": "cpp", |         "any": "cpp", | ||||||
|         "algorithm": "cpp", |         "array": "cpp", | ||||||
|         "numeric": "cpp", |         "atomic": "cpp", | ||||||
|         "ratio": "cpp", |         "strstream": "cpp", | ||||||
|         "utility": "cpp", |         "bit": "cpp", | ||||||
|         "iomanip": "cpp", |         "*.tcc": "cpp", | ||||||
|         "iosfwd": "cpp", |  | ||||||
|         "limits": "cpp", |  | ||||||
|         "numbers": "cpp", |  | ||||||
|         "cinttypes": "cpp", |  | ||||||
|         "bitset": "cpp", |         "bitset": "cpp", | ||||||
|  |         "charconv": "cpp", | ||||||
|  |         "chrono": "cpp", | ||||||
|  |         "codecvt": "cpp", | ||||||
|  |         "compare": "cpp", | ||||||
|  |         "complex": "cpp", | ||||||
|  |         "concepts": "cpp", | ||||||
|  |         "condition_variable": "cpp", | ||||||
|  |         "coroutine": "cpp", | ||||||
|  |         "cstdint": "cpp", | ||||||
|  |         "deque": "cpp", | ||||||
|  |         "forward_list": "cpp", | ||||||
|  |         "list": "cpp", | ||||||
|  |         "map": "cpp", | ||||||
|         "set": "cpp", |         "set": "cpp", | ||||||
|         "regex": "cpp" |         "string": "cpp", | ||||||
|  |         "unordered_map": "cpp", | ||||||
|  |         "vector": "cpp", | ||||||
|  |         "exception": "cpp", | ||||||
|  |         "algorithm": "cpp", | ||||||
|  |         "functional": "cpp", | ||||||
|  |         "iterator": "cpp", | ||||||
|  |         "memory": "cpp", | ||||||
|  |         "memory_resource": "cpp", | ||||||
|  |         "numeric": "cpp", | ||||||
|  |         "optional": "cpp", | ||||||
|  |         "random": "cpp", | ||||||
|  |         "ratio": "cpp", | ||||||
|  |         "source_location": "cpp", | ||||||
|  |         "string_view": "cpp", | ||||||
|  |         "system_error": "cpp", | ||||||
|  |         "tuple": "cpp", | ||||||
|  |         "type_traits": "cpp", | ||||||
|  |         "utility": "cpp", | ||||||
|  |         "fstream": "cpp", | ||||||
|  |         "future": "cpp", | ||||||
|  |         "initializer_list": "cpp", | ||||||
|  |         "iomanip": "cpp", | ||||||
|  |         "iostream": "cpp", | ||||||
|  |         "istream": "cpp", | ||||||
|  |         "limits": "cpp", | ||||||
|  |         "mutex": "cpp", | ||||||
|  |         "new": "cpp", | ||||||
|  |         "numbers": "cpp", | ||||||
|  |         "ostream": "cpp", | ||||||
|  |         "ranges": "cpp", | ||||||
|  |         "semaphore": "cpp", | ||||||
|  |         "shared_mutex": "cpp", | ||||||
|  |         "stdexcept": "cpp", | ||||||
|  |         "stop_token": "cpp", | ||||||
|  |         "streambuf": "cpp", | ||||||
|  |         "thread": "cpp", | ||||||
|  |         "cinttypes": "cpp", | ||||||
|  |         "typeindex": "cpp", | ||||||
|  |         "typeinfo": "cpp", | ||||||
|  |         "valarray": "cpp", | ||||||
|  |         "variant": "cpp", | ||||||
|  |         "*.ipp": "cpp", | ||||||
|  |         "format": "cpp", | ||||||
|  |         "span": "cpp", | ||||||
|  |         "__bit_reference": "cpp", | ||||||
|  |         "__bits": "cpp", | ||||||
|  |         "__config": "cpp", | ||||||
|  |         "__debug": "cpp", | ||||||
|  |         "__errc": "cpp", | ||||||
|  |         "__hash_table": "cpp", | ||||||
|  |         "__locale": "cpp", | ||||||
|  |         "__mutex_base": "cpp", | ||||||
|  |         "__node_handle": "cpp", | ||||||
|  |         "__nullptr": "cpp", | ||||||
|  |         "__split_buffer": "cpp", | ||||||
|  |         "__string": "cpp", | ||||||
|  |         "__threading_support": "cpp", | ||||||
|  |         "__tree": "cpp", | ||||||
|  |         "__tuple": "cpp", | ||||||
|  |         "ios": "cpp", | ||||||
|  |         "locale": "cpp", | ||||||
|  |         "queue": "cpp", | ||||||
|  |         "hash_map": "cpp", | ||||||
|  |         "hash_set": "cpp", | ||||||
|  |         "regex": "cpp", | ||||||
|  |         "stack": "cpp", | ||||||
|  |         "__memory": "cpp" | ||||||
|     } |     } | ||||||
| } | } | ||||||
							
								
								
									
										4
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							| @@ -8,7 +8,7 @@ | |||||||
|             "command": "make", |             "command": "make", | ||||||
|             "args": [], |             "args": [], | ||||||
|             "options": { |             "options": { | ||||||
|                 "cwd": "/home/happytanuki/BumbleCee/build/" |                 "cwd": "${workspaceFolder}/build/" | ||||||
|             }, |             }, | ||||||
|             "group": { |             "group": { | ||||||
|                 "kind": "build", |                 "kind": "build", | ||||||
| @@ -24,7 +24,7 @@ | |||||||
|                 ".." |                 ".." | ||||||
|             ], |             ], | ||||||
|             "options": { |             "options": { | ||||||
|                 "cwd": "/home/happytanuki/BumbleCee/build/" |                 "cwd": "${workspaceFolder}/build/" | ||||||
|             }, |             }, | ||||||
|             "group": { |             "group": { | ||||||
|                 "kind": "build", |                 "kind": "build", | ||||||
|   | |||||||
| @@ -5,31 +5,21 @@ set(BOT_NAME "BumbleCee") | |||||||
|  |  | ||||||
| project(${BOT_NAME}) | project(${BOT_NAME}) | ||||||
| aux_source_directory("src" coresrc) | aux_source_directory("src" coresrc) | ||||||
|  | aux_source_directory("src/Audio" commands) | ||||||
| aux_source_directory("src/Commands" commands) | aux_source_directory("src/Commands" commands) | ||||||
| add_executable(${BOT_NAME} ${coresrc} ${commands}) | aux_source_directory("src/Queue" settings) | ||||||
|  | aux_source_directory("src/Settings" settings) | ||||||
| string(ASCII 27 Esc) | aux_source_directory("src/Utils" settings) | ||||||
|  | add_executable(${BOT_NAME} ${coresrc} ${commands} ${settings}) | ||||||
|  |  | ||||||
| set(CMAKE_POSITION_INDEPENDENT_CODE ON) | set(CMAKE_POSITION_INDEPENDENT_CODE ON) | ||||||
|  | set(CMAKE_BUILD_TYPE Debug) | ||||||
| set_target_properties(${BOT_NAME} PROPERTIES | set(CMAKE_CXX_STANDARD 20) | ||||||
|     CXX_STANDARD 17 | set(CMAKE_CXX_STANDARD_REQUIRED ON) | ||||||
|     CXX_STANDARD_REQUIRED ON |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| set(THREADS_PREFER_PTHREAD_FLAG TRUE) | set(THREADS_PREFER_PTHREAD_FLAG TRUE) | ||||||
| find_package(Threads REQUIRED) | find_package(Threads REQUIRED) | ||||||
| find_package(DPP) | find_package(OpenSSL REQUIRED) | ||||||
| if(APPLE) |  | ||||||
| 	if(CMAKE_APPLE_SILICON_PROCESSOR) |  | ||||||
| 		set(OPENSSL_ROOT_DIR "/opt/homebrew/opt/openssl") |  | ||||||
| 	else() |  | ||||||
| 		set(OPENSSL_ROOT_DIR "/usr/local/opt/openssl") |  | ||||||
| 	endif() |  | ||||||
| 	find_package(OpenSSL REQUIRED) |  | ||||||
| else() |  | ||||||
| 	find_package(OpenSSL REQUIRED) |  | ||||||
| endif() |  | ||||||
|  |  | ||||||
| target_include_directories(${BOT_NAME} PUBLIC | target_include_directories(${BOT_NAME} PUBLIC | ||||||
|     ${CMAKE_CURRENT_SOURCE_DIR}/include |     ${CMAKE_CURRENT_SOURCE_DIR}/include | ||||||
| @@ -38,10 +28,8 @@ target_include_directories(${BOT_NAME} PUBLIC | |||||||
| ) | ) | ||||||
|  |  | ||||||
| target_link_libraries(${BOT_NAME} | target_link_libraries(${BOT_NAME} | ||||||
|     dl |  | ||||||
|     dpp |     dpp | ||||||
|     opus |     opus | ||||||
|     opusfile |  | ||||||
|     ogg |     ogg | ||||||
|     oggz |     oggz | ||||||
|     ${CMAKE_THREAD_LIBS_INIT} |     ${CMAKE_THREAD_LIBS_INIT} | ||||||
| @@ -49,32 +37,4 @@ target_link_libraries(${BOT_NAME} | |||||||
|     ${OPENSSL_SSL_LIBRARY} |     ${OPENSSL_SSL_LIBRARY} | ||||||
| ) | ) | ||||||
|  |  | ||||||
| if (DPP_FOUND) | link_directories(/usr/lib) | ||||||
|     target_link_libraries(${BOT_NAME} ${DPP_LIBRARIES}) |  | ||||||
|     target_include_directories(${BOT_NAME} PUBLIC ${DPP_INCLUDE_DIR}) |  | ||||||
| else() |  | ||||||
|     message(WARNING "Could not find DPP install. Building from source instead.") |  | ||||||
|     option(DPP_BUILD_TEST "" OFF) |  | ||||||
|     include(FetchContent) |  | ||||||
|  |  | ||||||
|     FetchContent_Declare( |  | ||||||
|             libdpp |  | ||||||
|             GIT_REPOSITORY https://github.com/brainboxdotcc/DPP.git |  | ||||||
|             GIT_TAG master) |  | ||||||
|  |  | ||||||
|     FetchContent_GetProperties(libdpp) |  | ||||||
|     if(NOT libdpp_POPULATED) |  | ||||||
|         FetchContent_Populate(libdpp) |  | ||||||
|         target_include_directories(${BOT_NAME} PUBLIC |  | ||||||
|             ${libdpp_SOURCE_DIR}/include |  | ||||||
|         ) |  | ||||||
|         add_subdirectory( |  | ||||||
|             ${libdpp_SOURCE_DIR} |  | ||||||
|             ${libdpp_BINARY_DIR} |  | ||||||
|             EXCLUDE_FROM_ALL) |  | ||||||
|     endif() |  | ||||||
|  |  | ||||||
|     target_link_libraries(${BOT_NAME} dpp) |  | ||||||
| endif() |  | ||||||
|  |  | ||||||
| set(CMAKE_CXX_FLAGS "-g") |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								DPP-markdown-logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								DPP-markdown-logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 5.6 KiB | 
							
								
								
									
										1
									
								
								Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | FROM alpine | ||||||
							
								
								
									
										38
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | |||||||
|  |  | ||||||
|  | # 이게 뭔가요? | ||||||
|  | C++ Dpp 라이브러리를 이용해서 개발된 간단한 디스코드 음악봇입니다! | ||||||
|  | <div align="center"> | ||||||
|  |   <a href="https://github.com/brainboxdotcc/DPP" alt="DPP"> <img src="DPP-markdown-logo.png" /> </a> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | # 어떻게 써요? | ||||||
|  | 1. 실행파일 경로에 config.json 파일을 만들고 다음과 같이 입력하세요: | ||||||
|  | ``` | ||||||
|  | {"token": "디스코드에서 발급받은 개인 봇 토큰"} | ||||||
|  | ``` | ||||||
|  | 2. Music 디렉터리를 만드세요 | ||||||
|  | 3. Music/Archive 파일을 만드세요 | ||||||
|  | 4. 봇을 실행시키고 초대하셔서 사용하시면 됩니다. | ||||||
|  |  | ||||||
|  | # 명령어 | ||||||
|  | ## /p | ||||||
|  | 노래를 예약합니다. | ||||||
|  | 사용법: | ||||||
|  | /p (노래링크 또는 노래 제목) | ||||||
|  | ## /q | ||||||
|  | 현재 큐의 내용물을 확인합니다. | ||||||
|  | 사용법: | ||||||
|  | /q | ||||||
|  | ## /d | ||||||
|  | 현재 큐의 내용물을 삭제합니다. | ||||||
|  | 현재 재생중인 곡은 0번입니다. | ||||||
|  | 사용법: | ||||||
|  | /d (큐의 노래번호) | ||||||
|  | ## /s | ||||||
|  | 현재 곡을 스킵합니다. | ||||||
|  | 사용법: | ||||||
|  | /s | ||||||
|  | ## /l | ||||||
|  | 음성 채팅을 떠납니다. | ||||||
|  | 사용법: | ||||||
|  | /l | ||||||
							
								
								
									
										69
									
								
								include/Audio/MusicPlayManager.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								include/Audio/MusicPlayManager.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | |||||||
|  | #pragma once | ||||||
|  | #ifndef _MUSICPLAYMANAGER_HPP_ | ||||||
|  | #define _MUSICPLAYMANAGER_HPP_ | ||||||
|  | #include <dpp/dpp.h> | ||||||
|  | #include <Queue/MusicQueue.hpp> | ||||||
|  | #include <condition_variable> | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  | class MusicPlayManager { | ||||||
|  | public: | ||||||
|  |     MusicPlayManager(std::shared_ptr<dpp::cluster> cluster, std::vector<dpp::snowflake> GIDs) : | ||||||
|  |         cluster(cluster), GIDs(GIDs) { | ||||||
|  |         queueMap = std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>>(); | ||||||
|  |         queueEmptyMutex = std::unordered_map<dpp::snowflake, std::shared_ptr<std::mutex>>(); | ||||||
|  |  | ||||||
|  |         cluster->on_voice_ready([this](const dpp::voice_ready_t &event){on_voice_ready(event);}); | ||||||
|  |         cluster->on_voice_track_marker([this](const dpp::voice_track_marker_t &event){on_voice_track_marker(event);}); | ||||||
|  |         cluster->on_voice_client_disconnect([this](const dpp::voice_client_disconnect_t& event){on_voice_client_disconnect(event);}); | ||||||
|  |  | ||||||
|  |         for (auto GID : GIDs) { | ||||||
|  |             queueMap[GID] = std::make_shared<MusicQueue>(); | ||||||
|  |             queueEmptyMutex[GID] = std::make_shared<std::mutex>(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @brief voice_ready 이벤트 인지시 콜백되는 함수 | ||||||
|  |      * @param event const dpp::voice_ready_t& | ||||||
|  |     **/ | ||||||
|  |     void on_voice_ready(const dpp::voice_ready_t& event); | ||||||
|  |     /** | ||||||
|  |      * @brief voice_track_marker 이벤트 인지시 콜백되는 함수 | ||||||
|  |      * @param event const dpp::voice_track_marker_t& | ||||||
|  |     **/ | ||||||
|  |     void on_voice_track_marker(const dpp::voice_track_marker_t& event); | ||||||
|  |     /** | ||||||
|  |      * @brief voice_client_disconnect 이벤트 인지시 콜백되는 함수 | ||||||
|  |      * @param event const dpp::voice_client_disconnect_t& | ||||||
|  |     **/ | ||||||
|  |     void on_voice_client_disconnect(const dpp::voice_client_disconnect_t& event); | ||||||
|  |  | ||||||
|  |     void play(dpp::discord_voice_client* client); | ||||||
|  |  | ||||||
|  |     void queue_music(const dpp::snowflake guildId, const std::shared_ptr<MusicQueueElement> music); | ||||||
|  |  | ||||||
|  |     void clear(const dpp::snowflake guildId); | ||||||
|  |     std::shared_ptr<MusicQueueElement> remove(const dpp::snowflake guildId, dpp::discord_voice_client* client, int index); | ||||||
|  |     int size(const dpp::snowflake guildId); | ||||||
|  |  | ||||||
|  |     void setRepeat(const dpp::snowflake guildId, const bool value); | ||||||
|  |     bool getRepeat(const dpp::snowflake guildId); | ||||||
|  |  | ||||||
|  |     std::list<MusicQueueElement> getQueue(const dpp::snowflake guildId); | ||||||
|  |     MusicQueueElement getNowPlaying(const dpp::snowflake guildId); | ||||||
|  |  | ||||||
|  |     std::condition_variable queuedCondition; | ||||||
|  | private: | ||||||
|  |     std::shared_ptr<dpp::cluster> cluster; | ||||||
|  |  | ||||||
|  |     std::vector<dpp::snowflake> GIDs; | ||||||
|  |     /// @brief 음악 큐 | ||||||
|  |     std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> queueMap; | ||||||
|  |  | ||||||
|  |     std::unordered_map<dpp::snowflake, std::shared_ptr<std::mutex>> queueEmptyMutex; | ||||||
|  |  | ||||||
|  |     void send_audio_to_voice(const MusicQueueElement& music, dpp::discord_voice_client* client); | ||||||
|  | }; | ||||||
|  | } | ||||||
|  | #endif | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <Commands/CommandType.hpp> |  | ||||||
| #include <dpp/dpp.h> |  | ||||||
| #include <memory> |  | ||||||
|  |  | ||||||
| class IBot { |  | ||||||
| public: |  | ||||||
|     IBot(std::string token, int totalShard); |  | ||||||
|     void start(); |  | ||||||
|     void onCommand(const dpp::slashcommand_t &event); |  | ||||||
|     void onReady(const dpp::ready_t &event); |  | ||||||
|  |  | ||||||
|     std::shared_ptr<dpp::cluster> botCluster; |  | ||||||
|     std::vector<std::shared_ptr<commands::ICommand>> commandsArray; |  | ||||||
| protected: |  | ||||||
| }; |  | ||||||
							
								
								
									
										53
									
								
								include/BumbleBee.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								include/BumbleBee.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | |||||||
|  | #pragma once | ||||||
|  | #ifndef _BUMBLEBEE_HPP_ | ||||||
|  | #define _BUMBLEBEE_HPP_ | ||||||
|  | #include <dpp/dpp.h> | ||||||
|  | #include <dpp/nlohmann/json.hpp> | ||||||
|  | #include <memory> | ||||||
|  | #include <Commands/BumbleBeeCommand.hpp> | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  | /** | ||||||
|  |  * @file BumbleBee.hpp | ||||||
|  |  * @brief 메인 봇 클래스 | ||||||
|  | **/ | ||||||
|  | class BumbleBee { | ||||||
|  | public: | ||||||
|  |     /** | ||||||
|  |      * @brief 생성자 | ||||||
|  |     **/ | ||||||
|  |     BumbleBee(); | ||||||
|  |     /** | ||||||
|  |      * @brief 파괴자 | ||||||
|  |      * @details BumbleBee의 모든 Property를 책임지고 파괴합니다 | ||||||
|  |     **/ | ||||||
|  |     ~BumbleBee() {} | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @brief 봇 시작 | ||||||
|  |     **/ | ||||||
|  |     void start(); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @brief slashcommand 이벤트 인지시 콜백되는 함수 | ||||||
|  |      * @param event const dpp::slashcommand_t& | ||||||
|  |     **/ | ||||||
|  |     void on_slashcommand(const dpp::slashcommand_t& event); | ||||||
|  |     /** | ||||||
|  |      * @brief ready 이벤트 인지시 콜백되는 함수 | ||||||
|  |      * @param event const dpp::ready_t& | ||||||
|  |     **/ | ||||||
|  |     void on_ready(const dpp::ready_t& event); | ||||||
|  |  | ||||||
|  |     /// @brief DPP 기본 클러스터 객체 | ||||||
|  |     std::shared_ptr<dpp::cluster> cluster; | ||||||
|  |     /// @brief guild id 배열 | ||||||
|  |     std::vector<dpp::snowflake> GIDs; | ||||||
|  | private: | ||||||
|  |     /// @brief Command 목록 | ||||||
|  |     std::unordered_map<std::string, std::shared_ptr<commands::ICommand>> commands; | ||||||
|  |     /// @brief voiceclient 관련 event 처리기 <guild id, 각 길드별 MusicPlayManager 인스턴스> | ||||||
|  |     std::shared_ptr<MusicPlayManager> musicManager; | ||||||
|  | }; | ||||||
|  | } | ||||||
|  | #endif | ||||||
| @@ -1,15 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <string> |  | ||||||
| #include <Bot.hpp> |  | ||||||
| #include <dpp/dpp.h> |  | ||||||
| #include <MusicQueue.hpp> |  | ||||||
| #include <memory> |  | ||||||
| #include <unordered_map> |  | ||||||
|  |  | ||||||
| class BumbleCeepp : public IBot { |  | ||||||
| public: |  | ||||||
|     BumbleCeepp(std::string token, int totalShard); |  | ||||||
| private: |  | ||||||
|     //<guild_id, queue> 쌍임. |  | ||||||
|     std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> queueMap; |  | ||||||
| }; |  | ||||||
							
								
								
									
										70
									
								
								include/Commands/BumbleBeeCommand.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								include/Commands/BumbleBeeCommand.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | |||||||
|  | #pragma once | ||||||
|  | #ifndef _BUMBLEBEECOMMAND_HPP_ | ||||||
|  | #define _BUMBLEBEECOMMAND_HPP_ | ||||||
|  | #include <dpp/dpp.h> | ||||||
|  | #include <Audio/MusicPlayManager.hpp> | ||||||
|  | #include <functional> | ||||||
|  |  | ||||||
|  | namespace bumbleBee::commands { | ||||||
|  | class ICommand : public dpp::slashcommand { | ||||||
|  | public: | ||||||
|  |     /** | ||||||
|  |      * @brief 기본 생성자 | ||||||
|  |      * @param botID 봇 아이디 | ||||||
|  |      * @param manager 음악재생 매니저 | ||||||
|  |     **/ | ||||||
|  |     ICommand(dpp::snowflake botID, std::shared_ptr<MusicPlayManager> manager) { | ||||||
|  |         this->botID = botID; | ||||||
|  |         this->musicManager = manager; | ||||||
|  |     } | ||||||
|  |     /** | ||||||
|  |      * @brief 명령어 호출 시에 콜백될 메소드 | ||||||
|  |      * @param event dpp::slashcommand_t | ||||||
|  |     **/ | ||||||
|  |     virtual void execute(const dpp::slashcommand_t &event){}; | ||||||
|  |  | ||||||
|  |     /// @brief 명령어 별명 | ||||||
|  |     std::vector<std::string> aliases; | ||||||
|  | private: | ||||||
|  |     /// @brief 봇 ID | ||||||
|  |     dpp::snowflake botID; | ||||||
|  | protected: | ||||||
|  |     /// @brief 음악재생 매니저 | ||||||
|  |     std::shared_ptr<MusicPlayManager> musicManager; | ||||||
|  |  | ||||||
|  | protected: | ||||||
|  |     /// @brief concrete class에서 구현해야 하는 init 이벤트트 | ||||||
|  |     virtual void init() = 0; | ||||||
|  | }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @brief 명령어 인자가 없는 명령어의 boilerplate 대체 매크로 | ||||||
|  |  * @param name 명령어 이름 및 클래스명 | ||||||
|  |  * @param description 명령어 설명 | ||||||
|  | **/ | ||||||
|  | #define _DECLARE_BUMBLEBEE_COMMAND(CLASS, NAME, DESCRIPTION) \ | ||||||
|  | namespace bumbleBee::commands { \ | ||||||
|  | class CLASS : public ICommand { \ | ||||||
|  | public: \ | ||||||
|  |     CLASS (dpp::snowflake botID, std::shared_ptr<MusicPlayManager> manager) \ | ||||||
|  |     : ICommand(botID, manager) { \ | ||||||
|  |         name = #NAME; \ | ||||||
|  |         description = DESCRIPTION; \ | ||||||
|  |         init(); \ | ||||||
|  |     } \ | ||||||
|  |     virtual void execute(const dpp::slashcommand_t &event) override; \ | ||||||
|  | protected: \ | ||||||
|  |     virtual void init() override; \ | ||||||
|  | }; \ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | _DECLARE_BUMBLEBEE_COMMAND(Delete,  d,          "큐의 해당하는 번호의 노래를 지웁니다") | ||||||
|  | _DECLARE_BUMBLEBEE_COMMAND(Leave,   l,          "음성 채팅방을 떠납니다") | ||||||
|  | _DECLARE_BUMBLEBEE_COMMAND(Play,    p,          "노래 재생") | ||||||
|  | _DECLARE_BUMBLEBEE_COMMAND(Queue,   q,          "노래 예약 큐를 확인합니다") | ||||||
|  | _DECLARE_BUMBLEBEE_COMMAND(Repeat,  r,          "반복을 켜거나 끕니다") | ||||||
|  | _DECLARE_BUMBLEBEE_COMMAND(Skip,    s,          "현재 재생중인 곡을 스킵합니다") | ||||||
|  | _DECLARE_BUMBLEBEE_COMMAND(Shuffle, shuffle,    "큐를 섞습니다") | ||||||
|  |  | ||||||
|  | #endif | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <dpp/dpp.h> |  | ||||||
| #include <vector> |  | ||||||
| #include <list> |  | ||||||
| #include <MusicQueue.hpp> |  | ||||||
|  |  | ||||||
| namespace commands { |  | ||||||
| class ICommand { |  | ||||||
| public: |  | ||||||
|     //이 생성자를 명시적으로 호출할 것. |  | ||||||
|     ICommand(std::shared_ptr<dpp::cluster> botCluster); |  | ||||||
|     virtual void operator()(const dpp::slashcommand_t &event) = 0; |  | ||||||
|  |  | ||||||
|     std::vector<dpp::slashcommand> commandObjectVector; |  | ||||||
| protected: |  | ||||||
|     std::shared_ptr<dpp::cluster> botCluster; |  | ||||||
| }; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| namespace commands { |  | ||||||
| class VCCommand : public ICommand { |  | ||||||
| public: |  | ||||||
|     VCCommand(std::shared_ptr<dpp::cluster> botCluster) : ICommand(botCluster) {} |  | ||||||
|  |  | ||||||
|     std::shared_ptr<MusicQueue> getQueue(const dpp::slashcommand_t& event); |  | ||||||
| protected: |  | ||||||
|     std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap; |  | ||||||
| }; |  | ||||||
| } |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <Commands/Play.hpp> |  | ||||||
| #include <Commands/Repeat.hpp> |  | ||||||
| #include <Commands/Queue.hpp> |  | ||||||
| #include <Commands/Skip.hpp> |  | ||||||
| #include <Commands/Leave.hpp> |  | ||||||
| #include <Commands/Delete.hpp> |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <Commands/CommandType.hpp> |  | ||||||
| #include <BumbleCeepp.hpp> |  | ||||||
| #include <memory> |  | ||||||
|  |  | ||||||
| namespace commands { |  | ||||||
| class Delete : public VCCommand { |  | ||||||
| public: |  | ||||||
|     Delete(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap); |  | ||||||
|  |  | ||||||
|     void operator()(const dpp::slashcommand_t& event); |  | ||||||
| }; |  | ||||||
| } |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <Commands/CommandType.hpp> |  | ||||||
| #include <BumbleCeepp.hpp> |  | ||||||
| #include <memory> |  | ||||||
|  |  | ||||||
| namespace commands { |  | ||||||
| class Leave : public VCCommand { |  | ||||||
| public: |  | ||||||
|     Leave(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap); |  | ||||||
|  |  | ||||||
|     void operator()(const dpp::slashcommand_t& event); |  | ||||||
| }; |  | ||||||
| } |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <Commands/CommandType.hpp> |  | ||||||
| #include <BumbleCeepp.hpp> |  | ||||||
| #include <memory> |  | ||||||
|  |  | ||||||
| namespace commands { |  | ||||||
| class Play : public VCCommand { |  | ||||||
| public: |  | ||||||
|     Play(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap); |  | ||||||
|  |  | ||||||
|     void operator()(const dpp::slashcommand_t& event); |  | ||||||
| }; |  | ||||||
| } |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <Commands/CommandType.hpp> |  | ||||||
| #include <BumbleCeepp.hpp> |  | ||||||
| #include <memory> |  | ||||||
|  |  | ||||||
| namespace commands { |  | ||||||
| class Queue : public VCCommand { |  | ||||||
| public: |  | ||||||
|     Queue(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap); |  | ||||||
|  |  | ||||||
|     void operator()(const dpp::slashcommand_t& event); |  | ||||||
| }; |  | ||||||
| } |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <Commands/CommandType.hpp> |  | ||||||
| #include <BumbleCeepp.hpp> |  | ||||||
| #include <memory> |  | ||||||
|  |  | ||||||
| namespace commands { |  | ||||||
| class Repeat : public VCCommand { |  | ||||||
| public: |  | ||||||
|     Repeat(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap); |  | ||||||
|  |  | ||||||
|     void operator()(const dpp::slashcommand_t& event); |  | ||||||
| }; |  | ||||||
| } |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <Commands/CommandType.hpp> |  | ||||||
| #include <BumbleCeepp.hpp> |  | ||||||
| #include <memory> |  | ||||||
|  |  | ||||||
| namespace commands { |  | ||||||
| class Skip : public VCCommand { |  | ||||||
| public: |  | ||||||
|     Skip(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap); |  | ||||||
|  |  | ||||||
|     void operator()(const dpp::slashcommand_t& event); |  | ||||||
| }; |  | ||||||
| } |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <string> |  | ||||||
| #include <dpp/dpp.h> |  | ||||||
|  |  | ||||||
| struct FQueueElement { |  | ||||||
|     std::string URL; |  | ||||||
|     std::string title; |  | ||||||
|     std::string description; |  | ||||||
|     std::string fileName; |  | ||||||
|     std::string thumbnail; |  | ||||||
|     std::string duration; |  | ||||||
|     dpp::embed embed; |  | ||||||
| }; |  | ||||||
| @@ -1,33 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| #include <list> |  | ||||||
| #include <FQueueElement.hpp> |  | ||||||
|  |  | ||||||
| struct FMusicQueueID { |  | ||||||
|     dpp::snowflake guild_id; |  | ||||||
|     uint32_t shard_id; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| class MusicQueue { |  | ||||||
| public: |  | ||||||
|     MusicQueue(FMusicQueueID id, std::shared_ptr<dpp::cluster> botCluster); |  | ||||||
|     void operator+=(FQueueElement operand); |  | ||||||
|     FQueueElement pop(int index); |  | ||||||
|     FQueueElement peek(int index); |  | ||||||
|     bool empty(); |  | ||||||
|     void clear(); |  | ||||||
|     std::list<struct FQueueElement>::iterator begin(); |  | ||||||
|     std::list<struct FQueueElement>::iterator end(); |  | ||||||
|     std::size_t size(); |  | ||||||
|     FMusicQueueID getId(); |  | ||||||
|     void play(); |  | ||||||
|  |  | ||||||
|     void markerCallback(); |  | ||||||
|  |  | ||||||
|     bool repeat; |  | ||||||
| private: |  | ||||||
|     std::list<struct FQueueElement> queue; |  | ||||||
|     std::mutex mutex; |  | ||||||
|     std::mutex playMutex; |  | ||||||
|     FMusicQueueID id; |  | ||||||
|     std::shared_ptr<dpp::cluster> botCluster; |  | ||||||
| }; |  | ||||||
							
								
								
									
										41
									
								
								include/Queue/MusicQueue.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								include/Queue/MusicQueue.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | |||||||
|  | #pragma once | ||||||
|  | #ifndef _MUSICQUEUE_HPP_ | ||||||
|  | #define _MUSICQUEUE_HPP_ | ||||||
|  | #include <memory> | ||||||
|  | #include <functional> | ||||||
|  | #include <condition_variable> | ||||||
|  | #include <list> | ||||||
|  | #include <dpp/dpp.h> | ||||||
|  | #include <Queue/MusicQueueElement.hpp> | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  |  | ||||||
|  | class MusicQueue { | ||||||
|  | public: | ||||||
|  |     MusicQueue() { | ||||||
|  |         queue = std::list<std::shared_ptr<MusicQueueElement>>(); | ||||||
|  |         currentPlayingPosition = queue.begin(); | ||||||
|  |         repeat = true; | ||||||
|  |     } | ||||||
|  |     void                                                    enqueue(std::shared_ptr<MusicQueueElement> Element); | ||||||
|  |     std::shared_ptr<MusicQueueElement>                      dequeue(); | ||||||
|  |     std::list<std::shared_ptr<MusicQueueElement>>::iterator findById(std::string id); | ||||||
|  |     std::list<std::shared_ptr<MusicQueueElement>>::iterator findByIndex(int index); | ||||||
|  |     std::shared_ptr<MusicQueueElement>                      nowplaying(); | ||||||
|  |     std::list<std::shared_ptr<MusicQueueElement>>::iterator next_music(); | ||||||
|  |     std::shared_ptr<MusicQueueElement>                      jump_to_index(int idx); | ||||||
|  |     void                                                    clear(); | ||||||
|  |     std::shared_ptr<MusicQueueElement>                      erase(std::list<std::shared_ptr<MusicQueueElement>>::iterator it); | ||||||
|  |     std::list<std::shared_ptr<MusicQueueElement>>           getQueueCopy(); | ||||||
|  |     int                                                     size(); | ||||||
|  |  | ||||||
|  |     bool repeat; | ||||||
|  |  | ||||||
|  |     std::list<std::shared_ptr<MusicQueueElement>>::iterator currentPlayingPosition; | ||||||
|  | private: | ||||||
|  |     std::list<std::shared_ptr<MusicQueueElement>> queue; | ||||||
|  |     std::mutex queueMutex; | ||||||
|  | }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #endif | ||||||
							
								
								
									
										23
									
								
								include/Queue/MusicQueueElement.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								include/Queue/MusicQueueElement.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | #pragma once | ||||||
|  | #ifndef _MUSICQUEUEELEMENT_HPP_ | ||||||
|  | #define _MUSICQUEUEELEMENT_HPP_ | ||||||
|  | #include <dpp/dpp.h> | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  |  | ||||||
|  | class MusicQueueElement { | ||||||
|  | public: | ||||||
|  |     MusicQueueElement( | ||||||
|  |         std::string id, | ||||||
|  |         std::string query, | ||||||
|  |         dpp::user issuingUser, | ||||||
|  |         dpp::embed embed) : | ||||||
|  |         id(id), query(query), issuingUser(issuingUser), embed(embed) {} | ||||||
|  |  | ||||||
|  |     const std::string id; | ||||||
|  |     const std::string query; | ||||||
|  |     const dpp::user issuingUser; | ||||||
|  |     const dpp::embed embed; | ||||||
|  | }; | ||||||
|  | } | ||||||
|  | #endif | ||||||
							
								
								
									
										37
									
								
								include/Settings/SettingsManager.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								include/Settings/SettingsManager.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | #pragma once | ||||||
|  | #ifndef _SETTINGSMANAGER_HPP_ | ||||||
|  | #define _SETTINGSMANAGER_HPP_ | ||||||
|  | #include <dpp/dpp.h> | ||||||
|  |  | ||||||
|  | #define _DECLARE_DEFAULT_ACCESSER_STATIC_VARIABLE(type, name, Name)\ | ||||||
|  | private:\ | ||||||
|  |     static type name;\ | ||||||
|  | public:\ | ||||||
|  |     static type get##Name() {return name;}\ | ||||||
|  |     static void set##Name(const type& value) {name = value; saveToFile();} | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  | /// @brief 모든 설정은 이 객체를 통해 스태틱하게 제공됨. | ||||||
|  | class SettingsManager { | ||||||
|  |     /// @brief 봇 토큰 | ||||||
|  |     _DECLARE_DEFAULT_ACCESSER_STATIC_VARIABLE(std::string, TOKEN, TOKEN) | ||||||
|  |     /// @brief yt-dlp 실행 명령어 | ||||||
|  |     _DECLARE_DEFAULT_ACCESSER_STATIC_VARIABLE(std::string, YTDLP_CMD, YTDLP_CMD) | ||||||
|  |     /// @brief ffmpeg 실행 명령어 | ||||||
|  |     _DECLARE_DEFAULT_ACCESSER_STATIC_VARIABLE(std::string, FFMPEG_CMD, FFMPEG_CMD) | ||||||
|  |     /// @brief 로그레벨 | ||||||
|  |     _DECLARE_DEFAULT_ACCESSER_STATIC_VARIABLE(dpp::loglevel, LOGLEVEL, LOGLEVEL) | ||||||
|  |     /// @brief 이전에 있던 명령을 지우고 등록할지 선택합니다. | ||||||
|  |     _DECLARE_DEFAULT_ACCESSER_STATIC_VARIABLE(bool, REGISTER_COMMAND, REGISTER_COMMAND) | ||||||
|  | public: | ||||||
|  |     /// @brief 설정 로드하기, 설정은 이 load()를 부르기 전까지는 적절하지 못한 값을 가지고 있음. | ||||||
|  |     /// @return 로드 성공 시 true, 실패 시 false 반환. | ||||||
|  |     static bool load(); | ||||||
|  |     /// @brief 설정 변경사항 저장 | ||||||
|  |     static void saveToFile(); | ||||||
|  |     /// @brief 토큰이 유효한지 체크합니다. | ||||||
|  |     /// @return 유효한 토큰이면 true, 아니면 false를 반환합니다. | ||||||
|  |     static bool validateToken(); | ||||||
|  | }; | ||||||
|  | } | ||||||
|  | #endif | ||||||
							
								
								
									
										60
									
								
								include/Utils/AsyncDownloadManager.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								include/Utils/AsyncDownloadManager.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | |||||||
|  | #pragma once | ||||||
|  | #ifndef _ASYNCDOWNLOADMANAGER_HPP_ | ||||||
|  | #define _ASYNCDOWNLOADMANAGER_HPP_ | ||||||
|  | #include <queue> | ||||||
|  | #include <string> | ||||||
|  | #include <thread> | ||||||
|  | #include <condition_variable> | ||||||
|  | #include <dpp/dpp.h> | ||||||
|  | #include <Queue/MusicQueue.hpp> | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  | /// @brief 싱글톤 멀티스레딩 다운로드 매니저 | ||||||
|  | class [[deprecated]] AsyncDownloadManager { | ||||||
|  | public: | ||||||
|  |     static AsyncDownloadManager& getInstance(int worker_count, std::weak_ptr<dpp::cluster> weak_cluster) { | ||||||
|  |         static AsyncDownloadManager dl(worker_count); | ||||||
|  |         dl.weak_cluster = weak_cluster; | ||||||
|  |         return dl; | ||||||
|  |     } | ||||||
|  |     void enqueue(std::pair<std::string, dpp::message> query) { | ||||||
|  |         std::thread th(&bumbleBee::AsyncDownloadManager::enqueueAsyncDL, this, query); | ||||||
|  |         th.detach(); | ||||||
|  |     } | ||||||
|  |     void stop() { | ||||||
|  |         auto cluster = weak_cluster.lock(); | ||||||
|  |         cluster->log(dpp::ll_info, "AsyncDownloadManager stop/destructor called."); | ||||||
|  |         terminate = true; | ||||||
|  |         dlQueueCondition.notify_all(); | ||||||
|  |  | ||||||
|  |         for (auto& t : worker_thread) { | ||||||
|  |             t.join(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     ~AsyncDownloadManager(){ | ||||||
|  |         stop(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     AsyncDownloadManager(int worker_count){ | ||||||
|  |         worker_thread.reserve(worker_count); | ||||||
|  |         terminate = false; | ||||||
|  |         for (int i=0; i<worker_count; i++) { | ||||||
|  |             worker_thread.emplace_back([this](){this->downloadWorker();}); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     void enqueueAsyncDL(std::pair<std::string, dpp::message> query); | ||||||
|  |     void downloadWorker(); | ||||||
|  |  | ||||||
|  |     std::queue<std::pair<std::string, dpp::message>> downloadQueue; | ||||||
|  |     std::condition_variable dlQueueCondition; | ||||||
|  |     std::mutex dlQueueMutex; | ||||||
|  |     std::weak_ptr<dpp::cluster> weak_cluster; | ||||||
|  |     std::vector<std::thread> worker_thread; | ||||||
|  |     bool terminate; | ||||||
|  | }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #endif | ||||||
							
								
								
									
										40
									
								
								include/Utils/ConsoleUtils.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								include/Utils/ConsoleUtils.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | #pragma once | ||||||
|  | #ifndef _CONSOLEUTILS_HPP_ | ||||||
|  | #define _CONSOLEUTILS_HPP_ | ||||||
|  | #include <iostream> | ||||||
|  | #include <sstream> | ||||||
|  | #include <queue> | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  | class ConsoleUtils { | ||||||
|  | public: | ||||||
|  |     /**  | ||||||
|  |      * @brief 명령어를 쉘에서 실행하고 결과를 EOF 전까지 읽어 \n을 구분자로 토큰화하여 반환합니다 | ||||||
|  |      * @param cmd 실행할 명령 | ||||||
|  |      * @return std::queue<std::string> tokens | ||||||
|  |      */ | ||||||
|  |     static std::queue<std::string> getResultFromCommand(std::string cmd) { | ||||||
|  |         std::string result, token; | ||||||
|  |         std::queue<std::string> tokens; | ||||||
|  |         FILE* stream; | ||||||
|  |         const int maxBuffer = 12; // 적당한 크기 | ||||||
|  |         char buffer[maxBuffer]; | ||||||
|  |         cmd.append(" 2>&1"); // 표준에러를 표준출력으로 redirect | ||||||
|  |  | ||||||
|  |         stream = popen(cmd.c_str(), "r"); // 주어진 command를 shell로 실행하고 파이프 연결 (fd 반환) | ||||||
|  |             if (stream) { | ||||||
|  |                 while (fgets(buffer, maxBuffer, stream) != NULL) result.append(buffer); // fgets: fd (stream)를 길이 (maxBuffer)만큼 읽어 버퍼 (buffer)에 저장 | ||||||
|  |                 pclose(stream); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         std::stringstream ss(result); | ||||||
|  |         while (std::getline(ss, token, '\n')) { | ||||||
|  |             tokens.push(token); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return tokens; | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #endif | ||||||
							
								
								
									
										77
									
								
								include/Utils/VersionsCheckUtils.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								include/Utils/VersionsCheckUtils.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | |||||||
|  | #pragma once | ||||||
|  | #ifndef _VERSIONCHECKUTILS_HPP_ | ||||||
|  | #define _VERSIONCHECKUTILS_HPP_ | ||||||
|  | #include <dpp/dpp.h> | ||||||
|  | #include "ConsoleUtils.hpp" | ||||||
|  | #include "../Settings/SettingsManager.hpp" | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  | class VersionsCheckUtils { | ||||||
|  | public: | ||||||
|  |     static bool isThereCMD(std::shared_ptr<dpp::cluster> cluster, std::string cmd) { | ||||||
|  |         if (ConsoleUtils::getResultFromCommand("which " + cmd).size() == 0) { | ||||||
|  |             cluster->log(dpp::ll_error, cmd + " is unavaliable. unresolable error please install " + cmd); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |             return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     static void validateYTDLPFFMPEGBinary(std::shared_ptr<dpp::cluster> cluster) { | ||||||
|  |         cluster->log(dpp::ll_info, "Checking if yt-dlp and ffmpeg is available..."); | ||||||
|  |         std::queue<std::string> result = ConsoleUtils::getResultFromCommand(SettingsManager::getFFMPEG_CMD() + " -version"); | ||||||
|  |         std::string front = result.front(); | ||||||
|  |         if (front[0] != 'f' || | ||||||
|  |             front[1] != 'f' || | ||||||
|  |             front[2] != 'm' || | ||||||
|  |             front[3] != 'p' || | ||||||
|  |             front[4] != 'e' || | ||||||
|  |             front[5] != 'g') { | ||||||
|  |             cluster->log(dpp::ll_warning, "ffmpeg is unavailable. downloading ffmpeg..."); | ||||||
|  |  | ||||||
|  |             if (!isThereCMD(cluster, "curl")) { | ||||||
|  |                 exit(1); | ||||||
|  |             } | ||||||
|  |             if (!isThereCMD(cluster, "tar")) { | ||||||
|  |                 exit(1); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             system("curl -LO https://github.com/BtbN/FFmpeg-Builds/releases/latest/download/ffmpeg-master-latest-linux64-gpl.tar.xz"); | ||||||
|  |             system("tar -xf ffmpeg-master-latest-linux64-gpl.tar.xz"); | ||||||
|  |             system("rm ffmpeg-master-latest-linux64-gpl.tar.xz"); | ||||||
|  |             system("mv ffmpeg-master-latest-linux64-gpl ffmpeg"); | ||||||
|  |             SettingsManager::setFFMPEG_CMD("./ffmpeg/bin/ffmpeg"); | ||||||
|  |         } | ||||||
|  |         result = ConsoleUtils::getResultFromCommand(SettingsManager::getYTDLP_CMD() + " --version"); | ||||||
|  |         front = result.front(); | ||||||
|  |         if ((front[0]-'0' < 0 || front[0]-'0' > 9) || | ||||||
|  |             (front[1]-'0' < 0 || front[1]-'0' > 9) || | ||||||
|  |             (front[2]-'0' < 0 || front[2]-'0' > 9) || | ||||||
|  |             (front[3]-'0' < 0 || front[3]-'0' > 9)) { | ||||||
|  |             cluster->log(dpp::ll_warning, "ytdlp is unavailable. downloading ytdlp..."); | ||||||
|  |  | ||||||
|  |             if (!isThereCMD(cluster, "curl")) { | ||||||
|  |                 exit(1); | ||||||
|  |             } | ||||||
|  |             if (!isThereCMD(cluster, "tar")) { | ||||||
|  |                 exit(1); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             system("curl -LO https://github.com/yt-dlp/yt-dlp/releases/latest/download/yt-dlp"); | ||||||
|  |             system("chmod +x ./yt-dlp"); | ||||||
|  |             SettingsManager::setYTDLP_CMD("./yt-dlp"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     static void updateytdlp(std::shared_ptr<dpp::cluster> cluster) { | ||||||
|  |         cluster->log(dpp::ll_info, "Checking if yt-dlp update is available..."); | ||||||
|  |         std::queue<std::string> result = ConsoleUtils::getResultFromCommand("./yt-dlp -U"); | ||||||
|  |         while(!result.empty()) { | ||||||
|  |             std::string front = result.front(); | ||||||
|  |             result.pop(); | ||||||
|  |             cluster->log(dpp::ll_info, front); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | } | ||||||
|  | #endif | ||||||
							
								
								
									
										125
									
								
								src/Audio/MusicPlayManager.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								src/Audio/MusicPlayManager.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,125 @@ | |||||||
|  | #include <Audio/MusicPlayManager.hpp> | ||||||
|  | #include <ogg/ogg.h> | ||||||
|  | #include <oggz/oggz.h> | ||||||
|  | #include <algorithm> | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  |  | ||||||
|  | void MusicPlayManager::on_voice_ready(const dpp::voice_ready_t& event) { | ||||||
|  |     play(event.voice_client); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void MusicPlayManager::on_voice_track_marker(const dpp::voice_track_marker_t& event) { | ||||||
|  |     play(event.voice_client); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void MusicPlayManager::on_voice_client_disconnect(const dpp::voice_client_disconnect_t& event) { // 안 불리는 듯? | ||||||
|  |     dpp::snowflake gid = dpp::find_channel(event.voice_client->channel_id)->guild_id; | ||||||
|  |     event.voice_client->stop_audio(); | ||||||
|  |     queueMap[gid]->clear(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void MusicPlayManager::play(dpp::discord_voice_client* client) { | ||||||
|  |     std::thread t([&](dpp::discord_voice_client* client){ | ||||||
|  |         dpp::snowflake gid = dpp::find_channel(client->channel_id)->guild_id; | ||||||
|  |  | ||||||
|  |         std::unique_lock<std::mutex> queueEmptyLock(*queueEmptyMutex[gid]); | ||||||
|  |         queuedCondition.wait(queueEmptyLock, [&]{ return queueMap[gid]->size() != 0; }); | ||||||
|  |  | ||||||
|  |         auto np = queueMap[gid]->next_music(); | ||||||
|  |         auto music = **np; | ||||||
|  |         send_audio_to_voice(music, client); | ||||||
|  |     }, client); | ||||||
|  |     t.detach(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void MusicPlayManager::queue_music(const dpp::snowflake guildId, const std::shared_ptr<MusicQueueElement> music) { | ||||||
|  |     queueMap[guildId]->enqueue(music); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void MusicPlayManager::clear(const dpp::snowflake guildId) { | ||||||
|  |     queueMap[guildId]->clear(); | ||||||
|  | } | ||||||
|  | std::shared_ptr<MusicQueueElement> MusicPlayManager::remove(const dpp::snowflake guildId, dpp::discord_voice_client* client, int index) { | ||||||
|  |     auto queue = queueMap[guildId]; | ||||||
|  |     auto foundIterator = queue->findByIndex(index); | ||||||
|  |  | ||||||
|  |     if (queue->currentPlayingPosition == foundIterator) { | ||||||
|  |         auto removed = queue->erase(queue->findByIndex(0)); | ||||||
|  |         if (client == nullptr) | ||||||
|  |             return removed; | ||||||
|  |         client->pause_audio(true); | ||||||
|  |         client->stop_audio(); | ||||||
|  |         client->pause_audio(false); | ||||||
|  |         client->insert_marker("end"); | ||||||
|  |         return removed; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return queue->erase(queue->findByIndex(index)); | ||||||
|  | } | ||||||
|  | int MusicPlayManager::size(const dpp::snowflake guildId) { | ||||||
|  |     return queueMap[guildId]->size(); | ||||||
|  | } | ||||||
|  | void MusicPlayManager::setRepeat(const dpp::snowflake guildId, const bool value) { | ||||||
|  |     queueMap[guildId]->repeat = value; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool MusicPlayManager::getRepeat(const dpp::snowflake guildId) { | ||||||
|  |     return queueMap[guildId]->repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | std::list<MusicQueueElement> MusicPlayManager::getQueue(const dpp::snowflake guildId){ | ||||||
|  |     std::list<std::shared_ptr<MusicQueueElement>> queue = queueMap[guildId]->getQueueCopy(); | ||||||
|  |     std::list<MusicQueueElement> returnValue; | ||||||
|  |  | ||||||
|  |     for (auto iter = queue.begin(); iter != queue.end(); iter++) | ||||||
|  |         returnValue.push_back(**iter); | ||||||
|  |  | ||||||
|  |     return returnValue; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | MusicQueueElement MusicPlayManager::getNowPlaying(const dpp::snowflake guildId) { | ||||||
|  |     std::shared_ptr<MusicQueueElement> nowplaying = queueMap[guildId]->nowplaying(); | ||||||
|  |     MusicQueueElement returnValue(*nowplaying); | ||||||
|  |     return returnValue; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void MusicPlayManager::send_audio_to_voice(const MusicQueueElement& music, dpp::discord_voice_client* client) { | ||||||
|  |     std::string command = "./streamOpus.sh ./yt-dlp ffmpeg https://youtu.be/"; | ||||||
|  |     command += music.id; | ||||||
|  |  | ||||||
|  |     OGGZ* og = oggz_open_stdio(popen(command.c_str(), "r"), OGGZ_READ); | ||||||
|  |  | ||||||
|  |     client->stop_audio(); | ||||||
|  |  | ||||||
|  |     oggz_set_read_callback( | ||||||
|  |         og, -1, | ||||||
|  |         [](OGGZ *oggz, oggz_packet *packet, long serialno, void *user_data) { | ||||||
|  |             auto voiceConn = (dpp::discord_voice_client *)user_data; | ||||||
|  |  | ||||||
|  |             voiceConn->send_audio_opus(packet->op.packet, packet->op.bytes); | ||||||
|  |  | ||||||
|  |             return 0; | ||||||
|  |         }, | ||||||
|  |         (void *)client | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     while (client && !client->terminating) { | ||||||
|  |         static const constexpr long CHUNK_READ = BUFSIZ * 2; | ||||||
|  |  | ||||||
|  |         const long read_bytes = oggz_read(og, CHUNK_READ); | ||||||
|  |  | ||||||
|  |         /* break on eof */ | ||||||
|  |         if (!read_bytes) { | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     client->creator->log(dpp::ll_info, "Sending " + music.id + " complete!"); | ||||||
|  |  | ||||||
|  |     oggz_close(og); | ||||||
|  |  | ||||||
|  |     client->insert_marker(); | ||||||
|  | } | ||||||
|  | } | ||||||
|  |  | ||||||
							
								
								
									
										42
									
								
								src/Bot.cpp
									
									
									
									
									
								
							
							
						
						
									
										42
									
								
								src/Bot.cpp
									
									
									
									
									
								
							| @@ -1,42 +0,0 @@ | |||||||
| #include <Bot.hpp> |  | ||||||
| #include <Commands/CommandType.hpp> |  | ||||||
|  |  | ||||||
| IBot::IBot(std::string token, int totalShard) |  | ||||||
| { |  | ||||||
|     botCluster = std::make_shared<dpp::cluster>(token, dpp::i_default_intents, totalShard); |  | ||||||
|     botCluster->on_log(dpp::utility::cout_logger()); |  | ||||||
|  |  | ||||||
|     botCluster->on_slashcommand([this](const dpp::slashcommand_t& event){onCommand(event);}); |  | ||||||
|     botCluster->on_ready([this](const dpp::ready_t &event){onReady(event);}); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void IBot::onCommand(const dpp::slashcommand_t &event) |  | ||||||
| { |  | ||||||
|     auto _event = event; |  | ||||||
|     for (int i = 0; i < commandsArray.size(); i++) { |  | ||||||
|         for (auto alias : commandsArray[i]->commandObjectVector) { |  | ||||||
|             if (event.command.get_command_name() == alias.name) { |  | ||||||
|                 (*commandsArray[i])(_event); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void IBot::onReady(const dpp::ready_t &event) |  | ||||||
| { |  | ||||||
|     if (!dpp::run_once<struct register_bot_commands>()) |  | ||||||
|         return; |  | ||||||
|  |  | ||||||
|     //BotCluster->global_bulk_command_delete(); |  | ||||||
|  |  | ||||||
|     for (int i = 0; i < commandsArray.size(); i++) { |  | ||||||
|         for (auto Alias : commandsArray[i]->commandObjectVector) { |  | ||||||
|             botCluster->global_command_create(Alias); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void IBot::start() |  | ||||||
| { |  | ||||||
|     botCluster->start(dpp::st_wait); |  | ||||||
| } |  | ||||||
							
								
								
									
										80
									
								
								src/BumbleBee.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								src/BumbleBee.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | |||||||
|  | #include <BumbleBee.hpp> | ||||||
|  | #include <memory> | ||||||
|  | #include <Settings/SettingsManager.hpp> | ||||||
|  | #include <Utils/VersionsCheckUtils.hpp> | ||||||
|  | #include <Audio/MusicPlayManager.hpp> | ||||||
|  | #include <Commands/BumbleBeeCommand.hpp> | ||||||
|  |  | ||||||
|  | namespace bumbleBee{ | ||||||
|  | BumbleBee::BumbleBee() { | ||||||
|  |     if (!SettingsManager::load()) { | ||||||
|  |         std::cout << "Please configure confing.json" << std::endl; | ||||||
|  |         exit(1); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     cluster = std::make_shared<dpp::cluster>(SettingsManager::getTOKEN()); | ||||||
|  |  | ||||||
|  |     cluster->on_log([](const dpp::log_t& event) { | ||||||
|  | 		if (event.severity >= SettingsManager::getLOGLEVEL()) { | ||||||
|  | 			std::cout << "[" << dpp::utility::current_date_time() << "] " << dpp::utility::loglevel(event.severity) << ": " << event.message << "\n"; | ||||||
|  | 		} | ||||||
|  | 	}); | ||||||
|  |     cluster->on_slashcommand([this](const dpp::slashcommand_t& event){on_slashcommand(event);}); | ||||||
|  |     cluster->on_ready([this](const dpp::ready_t &event){on_ready(event);}); | ||||||
|  |  | ||||||
|  |     VersionsCheckUtils::validateYTDLPFFMPEGBinary(cluster); | ||||||
|  |     VersionsCheckUtils::updateytdlp(cluster); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void BumbleBee::start() { this->cluster->start(dpp::st_wait); } | ||||||
|  |  | ||||||
|  | void BumbleBee::on_slashcommand(const dpp::slashcommand_t &event) { | ||||||
|  |     if (commands.find(event.command.get_command_name()) != commands.end()) { | ||||||
|  |         event.thinking(); | ||||||
|  |         auto command = commands.at(event.command.get_command_name()); | ||||||
|  |         std::thread t([](const dpp::slashcommand_t &event, std::shared_ptr<commands::ICommand> command){ | ||||||
|  |             command->execute(event); | ||||||
|  |         }, event, command); | ||||||
|  |         t.detach(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void BumbleBee::on_ready(const dpp::ready_t &event) { | ||||||
|  |     GIDs = event.guilds; | ||||||
|  |     musicManager = std::make_shared<MusicPlayManager>(cluster, GIDs); | ||||||
|  |  | ||||||
|  |     commands["d"]       = std::make_shared<commands::Delete>  (cluster->cluster_id, musicManager); | ||||||
|  |     commands["l"]       = std::make_shared<commands::Leave>   (cluster->cluster_id, musicManager); | ||||||
|  |     commands["p"]       = std::make_shared<commands::Play>    (cluster->cluster_id, musicManager); | ||||||
|  |     commands["q"]       = std::make_shared<commands::Queue>   (cluster->cluster_id, musicManager); | ||||||
|  |     commands["r"]       = std::make_shared<commands::Repeat>  (cluster->cluster_id, musicManager); | ||||||
|  |     commands["s"]       = std::make_shared<commands::Skip>    (cluster->cluster_id, musicManager); | ||||||
|  |     commands["shuffle"] = std::make_shared<commands::Shuffle> (cluster->cluster_id, musicManager); | ||||||
|  |      | ||||||
|  |     if (event.guilds.size() == 0) { | ||||||
|  |         cluster->log(dpp::loglevel::ll_info, "Bot is not on any server! Please invite this bot to any server."); | ||||||
|  |         exit(1); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     if (dpp::run_once<struct register_bot_commands>()) { | ||||||
|  |         if (SettingsManager::getREGISTER_COMMAND()) { | ||||||
|  |             cluster->log(dpp::loglevel::ll_info, "Clear Pre-installed commands"); | ||||||
|  |  | ||||||
|  |             cluster->global_bulk_command_create({ | ||||||
|  |                 *commands["d"], | ||||||
|  |                 *commands["l"], | ||||||
|  |                 *commands["p"], | ||||||
|  |                 *commands["q"], | ||||||
|  |                 *commands["r"], | ||||||
|  |                 *commands["s"], | ||||||
|  |                 *commands["shuffle"] | ||||||
|  |             }, [&](const dpp::confirmation_callback_t &t){ | ||||||
|  |                 cluster->log(dpp::loglevel::ll_info, "Command created."); | ||||||
|  |                 SettingsManager::setREGISTER_COMMAND(false); | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     cluster->log(dpp::loglevel::ll_info, "Bot ready."); | ||||||
|  | } | ||||||
|  | } | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| #include "BumbleCeepp.hpp" |  | ||||||
| #include <string> |  | ||||||
| #include <FQueueElement.hpp> |  | ||||||
| #include <Commands/Commands.hpp> |  | ||||||
|  |  | ||||||
| BumbleCeepp::BumbleCeepp(std::string token, int totalShard) |  | ||||||
|     : IBot(token, totalShard) |  | ||||||
| { |  | ||||||
|     commandsArray.push_back(std::make_shared<commands::Play>(botCluster, &queueMap)); |  | ||||||
|     commandsArray.push_back(std::make_shared<commands::Repeat>(botCluster, &queueMap)); |  | ||||||
|     commandsArray.push_back(std::make_shared<commands::Queue>(botCluster, &queueMap)); |  | ||||||
|     commandsArray.push_back(std::make_shared<commands::Skip>(botCluster, &queueMap)); |  | ||||||
|     commandsArray.push_back(std::make_shared<commands::Leave>(botCluster, &queueMap)); |  | ||||||
|     commandsArray.push_back(std::make_shared<commands::Delete>(botCluster, &queueMap)); |  | ||||||
|  |  | ||||||
|     std::cout << "Command added.\n"; |  | ||||||
| } |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| #include <Commands/CommandType.hpp> |  | ||||||
|  |  | ||||||
| commands::ICommand::ICommand(std::shared_ptr<dpp::cluster> botCluster) |  | ||||||
| { |  | ||||||
|     this->botCluster = botCluster; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| std::shared_ptr<MusicQueue> commands::VCCommand::getQueue(const dpp::slashcommand_t& event) { |  | ||||||
|     auto findResult = queueMap->find(event.command.guild_id); |  | ||||||
|     if (findResult == queueMap->end()) |  | ||||||
|     { |  | ||||||
|         FMusicQueueID queueID; |  | ||||||
|         queueID.guild_id = event.command.guild_id; |  | ||||||
|         queueID.shard_id = event.from->shard_id; |  | ||||||
|  |  | ||||||
|         (*queueMap)[queueID.guild_id] = std::make_shared<MusicQueue>(queueID, botCluster); |  | ||||||
|     } |  | ||||||
|     return queueMap->find(event.command.guild_id)->second; |  | ||||||
| } |  | ||||||
| @@ -1,62 +1,39 @@ | |||||||
| #include <Commands/Delete.hpp> | #include <Commands/BumbleBeeCommand.hpp> | ||||||
| #include <iostream> | #include <format> | ||||||
|  |  | ||||||
| commands::Delete::Delete(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap) | namespace bumbleBee::commands { | ||||||
|     : VCCommand(botCluster) |     void Delete::execute(const dpp::slashcommand_t &event) { | ||||||
| { |         if (std::holds_alternative<std::monostate>(event.get_parameter("pos"))) // 여기 들어올 일 있나? | ||||||
|     this->queueMap = queueMap; |         { | ||||||
|     dpp::slashcommand Command = dpp::slashcommand("d", "큐의 해당하는 번호의 노래를 지웁니다", botCluster->me.id); |             event.edit_original_response(dpp::message("위치를 제공하여 주십시오")); | ||||||
|  |             event.reply(""); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         int pos = std::get<std::int64_t>(event.get_parameter("pos")); | ||||||
|  |  | ||||||
|     Command.add_option( |         if (pos < 0 || pos > musicManager->size(event.command.guild_id)) | ||||||
|         dpp::command_option(dpp::co_string, "pos", "큐 번호", botCluster->me.id) |         { | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     commandObjectVector.push_back(Command); |             event.edit_original_response(dpp::message(std::string("이상한 인덱스 위치. Pos :") + std::to_string(pos))); | ||||||
| } |  | ||||||
|  |  | ||||||
| void commands::Delete::operator()(const dpp::slashcommand_t& event) |  | ||||||
| { |  | ||||||
|     if (std::holds_alternative<std::monostate>(event.get_parameter("pos"))) { |  | ||||||
|         event.reply("삭제할 노래의 인덱스가 정확하지 않습니다."); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|     std::string Pos = std::get<std::string>(event.get_parameter("pos")); |  | ||||||
|     event.thinking(); |  | ||||||
|  |  | ||||||
|     std::shared_ptr<MusicQueue> queue = getQueue(event); |  | ||||||
|  |  | ||||||
|     auto index = atoi(Pos.c_str()); |  | ||||||
|  |  | ||||||
|     int queueSize = queue->size(); |  | ||||||
|  |  | ||||||
|     std::cout << "queue size : " << queueSize << "\n"; |  | ||||||
|  |  | ||||||
|     if (index < 0 || (queueSize - 1) < index) { |  | ||||||
|         std::cout << "invalid index : " << index << ", " + Pos + "\n"; |  | ||||||
|  |  | ||||||
|         event.edit_original_response(dpp::message(event.command.channel_id, "이상한 인덱스 위치. Pos : " + Pos)); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     auto PopedElement = queue->pop(index); |  | ||||||
|  |  | ||||||
|     dpp::embed embed = PopedElement.embed |  | ||||||
|         .set_timestamp(time(0)); |  | ||||||
|  |  | ||||||
|     dpp::message msg(event.command.channel_id, "다음 항목을 큐에서 삭제했습니다!:"); |  | ||||||
|  |  | ||||||
|     if (atoi(Pos.c_str()) == 0) { |  | ||||||
|         dpp::voiceconn* v = event.from->get_voice(event.command.guild_id); |  | ||||||
|  |  | ||||||
|         if (!v || !v->voiceclient || !v->voiceclient->is_ready()) { |  | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         v->voiceclient->stop_audio(); |         dpp::voiceconn* v = event.from->get_voice(event.command.guild_id); | ||||||
|         v->voiceclient->insert_marker("end of music"); |  | ||||||
|     } |  | ||||||
|      |  | ||||||
|     msg.add_embed(embed); |  | ||||||
|  |  | ||||||
|     event.edit_original_response(msg); |         std::shared_ptr<MusicQueueElement> removed; | ||||||
|  |          | ||||||
|  |         if (!v) // v-> 로 nullptr을 참조하면 안 되므로. | ||||||
|  |             removed = musicManager->remove(event.command.guild_id, nullptr, pos); | ||||||
|  |         else | ||||||
|  |             removed = musicManager->remove(event.command.guild_id, v->voiceclient, pos); | ||||||
|  |  | ||||||
|  |         dpp::message msg("다음 항목을 큐에서 삭제했습니다!:"); | ||||||
|  |         msg.add_embed(removed->embed); | ||||||
|  |  | ||||||
|  |         event.edit_original_response(msg); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void Delete::init() { | ||||||
|  |         add_option(dpp::command_option(dpp::co_integer, "pos", "지울 위치(위치는 1부터 시작, 현재 재생곡은 0)", true)); | ||||||
|  |     } | ||||||
| } | } | ||||||
| @@ -1,30 +1,19 @@ | |||||||
| #include <Commands/Leave.hpp> | #include <Commands/BumbleBeeCommand.hpp> | ||||||
| #include <iostream> |  | ||||||
|  |  | ||||||
| commands::Leave::Leave(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap) | namespace bumbleBee::commands { | ||||||
|     : VCCommand(botCluster) |     void Leave::execute(const dpp::slashcommand_t &event) { // 왜 read loop ended가 뜨는가... | ||||||
| { |         dpp::voiceconn* v = event.from->get_voice(event.command.guild_id); | ||||||
|     this->queueMap = queueMap; |  | ||||||
|     dpp::slashcommand command = dpp::slashcommand("l", "음챗을 떠납니다", botCluster->me.id); |  | ||||||
|  |  | ||||||
|     commandObjectVector.push_back(command); |         if (!v || !v->voiceclient || !v->voiceclient->is_ready()) { | ||||||
| } |             return; | ||||||
|  |         } | ||||||
|  |         musicManager->clear(event.command.guild_id); | ||||||
|  |          | ||||||
|  |         event.from->disconnect_voice(event.command.guild_id); | ||||||
|  |  | ||||||
| void commands::Leave::operator()(const dpp::slashcommand_t& event) |         event.edit_original_response(dpp::message("음성 채팅방을 떠납니다!")); | ||||||
| { |  | ||||||
|     dpp::voiceconn* v = event.from->get_voice(event.command.guild_id); |  | ||||||
|  |  | ||||||
|     if (!v || !v->voiceclient || !v->voiceclient->is_ready()) { |  | ||||||
|         return; |  | ||||||
|     } |     } | ||||||
|     v->voiceclient->stop_audio(); |  | ||||||
|  |  | ||||||
|     std::shared_ptr<MusicQueue> queue = getQueue(event); |     void Leave::init() { | ||||||
|  |     } | ||||||
|     queue->clear(); |  | ||||||
|     event.from->disconnect_voice(event.command.guild_id); |  | ||||||
|  |  | ||||||
|     dpp::message msg(event.command.channel_id, "음성 채팅방을 떠납니다!"); |  | ||||||
|  |  | ||||||
|     event.reply(msg); |  | ||||||
| } | } | ||||||
| @@ -1,126 +1,167 @@ | |||||||
| #include <Commands/Play.hpp> | #include <Commands/BumbleBeeCommand.hpp> | ||||||
| #include <dpp/dpp.h> | #include <Utils/ConsoleUtils.hpp> | ||||||
|  | #include <Settings/SettingsManager.hpp> | ||||||
| #include <dpp/nlohmann/json.hpp> | #include <dpp/nlohmann/json.hpp> | ||||||
| #include <string> | #include <variant> | ||||||
| #include <filesystem> |  | ||||||
| #include <ctime> |  | ||||||
|  |  | ||||||
| using json = nlohmann::json; | namespace bumbleBee::commands { | ||||||
|  |     void Play::execute(const dpp::slashcommand_t &event) { | ||||||
|  |         dpp::guild *g = dpp::find_guild(event.command.guild_id); | ||||||
|  |  | ||||||
| commands::Play::Play(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap) |         if (!g) { //wtf? | ||||||
|     : VCCommand(botCluster) |             event.edit_original_response(dpp::message("GUILD NOT FOUND!! WHAT IS THIS SORCERY??")); | ||||||
| { |             return; | ||||||
|     this->queueMap = queueMap; |  | ||||||
|     dpp::slashcommand command = dpp::slashcommand("p", "노래 재생", botCluster->me.id); |  | ||||||
|  |  | ||||||
|     command.add_option( |  | ||||||
|         dpp::command_option(dpp::co_string, "query", "링크 또는 검색어", botCluster->me.id) |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     commandObjectVector.push_back(command); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void commands::Play::operator()(const dpp::slashcommand_t& event) |  | ||||||
| { |  | ||||||
|     if (std::holds_alternative<std::monostate>(event.get_parameter("query"))) |  | ||||||
|     { |  | ||||||
|         event.reply("노래를 재생하려면 검색어 또는 링크를 입력해 주십시오."); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /* Attempt to connect to a voice channel, returns false if we fail to connect. */ |  | ||||||
|     if (!event.from->get_voice(event.command.guild_id)) |  | ||||||
|     { |  | ||||||
|         if (!dpp::find_guild(event.command.guild_id)->connect_member_voice(event.command.get_issuing_user().id)) |  | ||||||
|         { |  | ||||||
|             return event.reply("노래를 재생할 음성 채팅방에 먼저 참가하고 신청해야 합니다!"); |  | ||||||
|         } |         } | ||||||
|     } |         if (std::holds_alternative<std::monostate>(event.get_parameter("query"))) // 이거 필요한가??? | ||||||
|  |         { | ||||||
|  |             event.reply("노래를 재생하려면 검색어 또는 링크를 입력해 주십시오."); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (!event.from->get_voice(event.command.guild_id) && !g->connect_member_voice(event.command.usr.id)) { | ||||||
|  |             event.edit_original_response(dpp::message("노래를 재생할 음성 채팅방에 먼저 참가하고 신청해야 합니다!")); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         std::string query = std::get<std::string>(event.get_parameter("query")); | ||||||
|  |  | ||||||
|     std::string Query = std::get<std::string>(event.get_parameter("query")); |         std::queue<std::string> ids = ConsoleUtils::getResultFromCommand(SettingsManager::getYTDLP_CMD() + " --default-search ytsearch --flat-playlist --skip-download --quiet --ignore-errors --print id " + query); | ||||||
|  |  | ||||||
|     event.thinking(); |         std::queue<std::shared_ptr<MusicQueueElement>> musics; | ||||||
|  |  | ||||||
|     std::shared_ptr<MusicQueue> queue = getQueue(event); |         int initialQueuedCount = musicManager->size(event.command.guild_id); | ||||||
|  |  | ||||||
|     std::cout << "다운로드 시작" << "\n"; |         dpp::message msg; | ||||||
|     std::system(("python3 yt-download.py \"" + Query + "\" & wait").c_str()); |  | ||||||
|     std::cout << "다운로드 완료" << "\n"; |  | ||||||
|  |  | ||||||
|     dpp::message msg(event.command.channel_id, "큐에 다음 곡을 추가했습니다:"); |         if (musicManager->size(event.command.guild_id) == 0) | ||||||
|  |             msg.content = "다음 곡을 재생합니다:"; | ||||||
|  |         else | ||||||
|  |             msg.content = "큐에 다음 곡을 추가했습니다:"; | ||||||
|  |  | ||||||
|     std::ifstream infofile, idfile; |         if (ids.size() >= 2) { | ||||||
|     json document; |             event.from->creator->log(dpp::ll_info, "Playlist detected."); | ||||||
|     std::string ID; |             while (!ids.empty()) { | ||||||
|     std::queue<FQueueElement> RequestedMusic; |                 if (ids.front() == "") { | ||||||
|     idfile.open("Temp/CurMusic"); |                     ids.pop(); | ||||||
|     while (std::getline(idfile, ID)) |                     continue; | ||||||
|     { |                 } | ||||||
|         std::cout << ID << "\n"; |  | ||||||
|         infofile.open("Music/" + ID + ".info.json"); |  | ||||||
|         infofile >> document; |  | ||||||
|         infofile.close(); |  | ||||||
|  |  | ||||||
|         time_t SongLength = int(document["duration"]); |                 FILE* file = popen((SettingsManager::getYTDLP_CMD() + " --default-search ytsearch --flat-playlist --skip-download --quiet --ignore-errors -J http://youtu.be/" + ids.front()).c_str(), "r"); | ||||||
|         char SongLengthStr[10]; |              | ||||||
|         tm t; |                 std::ostringstream oss; | ||||||
|         t.tm_mday = SongLength / 86400; |                 char buffer[1024]; | ||||||
|         t.tm_hour = (SongLength % 86400)/3600; |                 size_t bytesRead; | ||||||
|         t.tm_min = (SongLength % 3600)/60; |                  | ||||||
|         t.tm_sec = SongLength%60; |                 while ((bytesRead = fread(buffer, 1, sizeof(buffer), file)) > 0) { | ||||||
|         strftime(SongLengthStr, sizeof(SongLengthStr), "%X", &t); |                     oss.write(buffer, bytesRead); | ||||||
|  |                 } | ||||||
|  |                 fclose(file); | ||||||
|  |  | ||||||
|         FQueueElement Data = { |                 std::istringstream iss(oss.str()); | ||||||
|             std::string(document["webpage_url"]), |                 nlohmann::json videoDataJson; | ||||||
|             std::string(document["title"]), |                 iss >> videoDataJson; | ||||||
|             std::string(document["uploader"]), |  | ||||||
|             std::string(document["id"]), |                 // std::string dump = videoDataJson.dump(4); | ||||||
|             std::string(document["thumbnail"]), |  | ||||||
|             to_string(document["duration"]), |                 time_t SongLength = int(videoDataJson["duration"]); | ||||||
|             dpp::embed() |                 char SongLengthStr[10]; | ||||||
|  |                 tm t; | ||||||
|  |                 t.tm_mday = SongLength / 86400; | ||||||
|  |                 t.tm_hour = (SongLength % 86400)/3600; | ||||||
|  |                 t.tm_min = (SongLength % 3600)/60; | ||||||
|  |                 t.tm_sec = SongLength%60; | ||||||
|  |                 strftime(SongLengthStr, sizeof(SongLengthStr), "%X", &t); | ||||||
|  |  | ||||||
|  |                 dpp::embed embed = dpp::embed() | ||||||
|  |                     .set_color(dpp::colors::sti_blue) | ||||||
|  |                     .set_title(std::string(videoDataJson["title"])) | ||||||
|  |                     .set_description(std::string(videoDataJson["uploader"])) | ||||||
|  |                     .set_url(std::string(videoDataJson["webpage_url"])) | ||||||
|  |                     .set_image(std::string(videoDataJson["thumbnail"])) | ||||||
|  |                     .add_field( | ||||||
|  |                         "길이", | ||||||
|  |                         SongLengthStr, | ||||||
|  |                         true | ||||||
|  |                     ); | ||||||
|  |  | ||||||
|  |                 musics.push(std::make_shared<MusicQueueElement>(ids.front(), query, event.command.usr, embed)); | ||||||
|  |                 ids.pop(); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (!ids.empty()) { | ||||||
|  |             FILE* file = popen((SettingsManager::getYTDLP_CMD() + " --default-search ytsearch --flat-playlist --skip-download --quiet --ignore-errors -J http://youtu.be/" + ids.front()).c_str(), "r"); | ||||||
|  |              | ||||||
|  |             std::ostringstream oss; | ||||||
|  |             char buffer[1024]; | ||||||
|  |             size_t bytesRead; | ||||||
|  |              | ||||||
|  |             while ((bytesRead = fread(buffer, 1, sizeof(buffer), file)) > 0) { | ||||||
|  |                 oss.write(buffer, bytesRead); | ||||||
|  |             } | ||||||
|  |             fclose(file); | ||||||
|  |  | ||||||
|  |             std::istringstream iss(oss.str()); | ||||||
|  |             nlohmann::json videoDataJson; | ||||||
|  |             iss >> videoDataJson; | ||||||
|  |  | ||||||
|  |             time_t SongLength = int(videoDataJson["duration"]); | ||||||
|  |             char SongLengthStr[10]; | ||||||
|  |             tm t; | ||||||
|  |             t.tm_mday = SongLength / 86400; | ||||||
|  |             t.tm_hour = (SongLength % 86400)/3600; | ||||||
|  |             t.tm_min = (SongLength % 3600)/60; | ||||||
|  |             t.tm_sec = SongLength%60; | ||||||
|  |             strftime(SongLengthStr, sizeof(SongLengthStr), "%X", &t); | ||||||
|  |  | ||||||
|  |             dpp::embed embed = dpp::embed() | ||||||
|                 .set_color(dpp::colors::sti_blue) |                 .set_color(dpp::colors::sti_blue) | ||||||
|                 .set_title(Data.title) |                 .set_title(std::string(videoDataJson["title"])) | ||||||
|                 .set_description(Data.description) |                 .set_description(std::string(videoDataJson["uploader"])) | ||||||
|                 .set_url(Data.URL) |                 .set_url(std::string(videoDataJson["webpage_url"])) | ||||||
|                 .set_image(Data.thumbnail) |                 .set_image(std::string(videoDataJson["thumbnail"])) | ||||||
|                 .add_field( |                 .add_field( | ||||||
|                     "길이", |                     "길이", | ||||||
|                     SongLengthStr, |                     SongLengthStr, | ||||||
|                     true |                     true | ||||||
|                 ) |                 ); | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         (*queue) += Data; |             musics.push(std::make_shared<MusicQueueElement>(ids.front(), query, event.command.usr, embed)); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         RequestedMusic.push(Data); |         if (musics.size() == 1) { | ||||||
|     } |             event.from->creator->log(dpp::ll_info, "Enqueuing " + musics.front()->id); | ||||||
|     idfile.close(); |             musicManager->queue_music(event.command.guild_id, musics.front()); | ||||||
|     std::system("rm -f Temp/CurMusic"); |             msg.add_embed(musics.front()->embed); | ||||||
|     std::cout << "queued\n"; |             musics.pop(); | ||||||
|  |              | ||||||
|  |             event.edit_original_response(msg); | ||||||
|  |             musicManager->queuedCondition.notify_all(); | ||||||
|  |         } | ||||||
|  |         else if (musics.size() > 1) { | ||||||
|  |             event.from->creator->log(dpp::ll_info, "Enqueuing " + musics.front()->id); | ||||||
|  |             musicManager->queue_music(event.command.guild_id, musics.front()); | ||||||
|  |             msg.add_embed(musics.front()->embed); | ||||||
|  |             musics.pop(); | ||||||
|  |  | ||||||
|     msg.add_embed(RequestedMusic.front().embed); |             event.edit_original_response(msg); | ||||||
|     RequestedMusic.pop(); |             musicManager->queuedCondition.notify_all(); | ||||||
|     event.edit_original_response(msg); |  | ||||||
|  |  | ||||||
|     while (!RequestedMusic.empty()) |             while (!musics.empty()) { | ||||||
|     { |                 event.from->creator->log(dpp::ll_info, "Enqueuing " + musics.front()->id); | ||||||
|         dpp::message followMsg(event.command.channel_id, ""); |                 dpp::message followMsg(event.command.channel_id, ""); | ||||||
|  |  | ||||||
|         followMsg.add_embed(RequestedMusic.front().embed); |                 followMsg.add_embed(musics.front()->embed); | ||||||
|         RequestedMusic.pop(); |                 event.from->creator->message_create(followMsg); // 어차피 원래 메시지를 지정해서 수정할 것이기 때문에 먼저 팔로잉 메시지를 작성해도 상관없음. | ||||||
|  |  | ||||||
|         botCluster->message_create(followMsg); |                 musicManager->queue_music(event.command.guild_id, musics.front()); | ||||||
|     } |                 musics.pop(); | ||||||
|     std::cout << "replied\n"; |             } | ||||||
|  |         } | ||||||
|     dpp::voiceconn* v = event.from->get_voice(event.command.guild_id); |         else { // ?? | ||||||
|  |             event.from->creator->log(dpp::ll_error, "??? not queueed any music"); | ||||||
|     /* If the voice channel was invalid, or there is an issue with it, then tell the user. */ |         } | ||||||
|     if (v && v->voiceclient && v->voiceclient->is_ready()) |  | ||||||
|     { |  | ||||||
|         queue->play(); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     botCluster->on_voice_ready([this, queue](const dpp::voice_ready_t& Voice){ queue->play(); }); |     void Play::init() { | ||||||
|     return; |         add_option(dpp::command_option(dpp::co_string, "query", "링크 또는 검색어", true)); | ||||||
|  |     } | ||||||
| } | } | ||||||
| @@ -1,97 +1,34 @@ | |||||||
| #include <Commands/Queue.hpp> | #include <Commands/BumbleBeeCommand.hpp> | ||||||
| #include <iostream> |  | ||||||
| #include <sstream> |  | ||||||
| #include <cmath> |  | ||||||
|  |  | ||||||
| namespace commands { | namespace bumbleBee::commands { | ||||||
| dpp::embed makeEmbed(std::list<FQueueElement>::iterator& iter, std::list<FQueueElement>::iterator end, bool Repeat = false, int Index = 0) |     void Queue::execute(const dpp::slashcommand_t &event) { | ||||||
| { |         auto queue = musicManager->getQueue(event.command.guild_id); | ||||||
|     dpp::embed embed = dpp::embed() |         auto nowplaying = musicManager->getNowPlaying(event.command.guild_id); | ||||||
|         .set_color(dpp::colors::sti_blue); |  | ||||||
|  |  | ||||||
|     if (iter == end) { |         dpp::message msg; | ||||||
|         embed |         dpp::embed embed; | ||||||
|             .set_title("큐가 비었습니다!") |  | ||||||
|             .set_timestamp(time(0)); |  | ||||||
|  |  | ||||||
|         if (Repeat) |         if (queue.size() == 0) { | ||||||
|             embed.add_field(":repeat:",""); |             embed | ||||||
|  |                 .set_title("큐가 비었습니다!") | ||||||
|         return embed; |                 .set_timestamp(time(0)); | ||||||
|     } |             msg.add_embed(embed); | ||||||
|  |             event.edit_original_response(msg); | ||||||
|     std::ostringstream Number; |             return; | ||||||
|     int Start = Index; |  | ||||||
|  |  | ||||||
|     for (; (Index < Start + 5) && (iter != end); iter++, Index++) { |  | ||||||
|         Number.clear(); |  | ||||||
|         Number.str(""); |  | ||||||
|         Number << Index; |  | ||||||
|         embed.add_field( |  | ||||||
|             Number.str(), |  | ||||||
|             "", |  | ||||||
|             true |  | ||||||
|         ) |  | ||||||
|         .add_field( |  | ||||||
|             iter->title, |  | ||||||
|             iter->description, |  | ||||||
|             true |  | ||||||
|         ) |  | ||||||
|         .add_field( |  | ||||||
|             "", |  | ||||||
|             "" |  | ||||||
|         ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (iter == end) { |  | ||||||
|         embed.set_timestamp(time(0)); |  | ||||||
|         if (Repeat) |  | ||||||
|             embed.add_field(":repeat:",""); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return embed; |  | ||||||
| } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| commands::Queue::Queue(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap) |  | ||||||
|     : VCCommand(botCluster) |  | ||||||
| { |  | ||||||
|     this->queueMap = queueMap; |  | ||||||
|     dpp::slashcommand command = dpp::slashcommand("q", "노래 예약 큐 확인", botCluster->me.id); |  | ||||||
|  |  | ||||||
|     commandObjectVector.push_back(command); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void commands::Queue::operator()(const dpp::slashcommand_t& event) { |  | ||||||
|     dpp::message msg; |  | ||||||
|     msg.set_channel_id(event.command.channel_id); |  | ||||||
|     std::shared_ptr<MusicQueue> queue = getQueue(event); |  | ||||||
|  |  | ||||||
|     if (queue->size() < 1) { |  | ||||||
|         auto iter = queue->begin(); |  | ||||||
|         msg.add_embed(makeEmbed(iter, queue->end(), queue->repeat)); |  | ||||||
|     } |  | ||||||
|     else { |  | ||||||
|         msg.set_content("지금 재생 중:"); |  | ||||||
|         msg.add_embed(queue->peek(0).embed); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     event.reply(msg, [this, queue, event](const dpp::confirmation_callback_t &_event) { |  | ||||||
|         auto iter = queue->begin(); |  | ||||||
|         int queueSize = queue->size(); |  | ||||||
|         iter++; |  | ||||||
|         for (int i = 0; i < ceil(queueSize / 5.0); i++) { |  | ||||||
|             dpp::embed followEmbed = makeEmbed(iter, queue->end(), queue->repeat, i * 5 + 1); |  | ||||||
|  |  | ||||||
|             dpp::message followMsg; |  | ||||||
|             followMsg.channel_id = event.command.channel_id; |  | ||||||
|  |  | ||||||
|             if (i == 0) { |  | ||||||
|                 followMsg.content = "현재 큐에 있는 항목:"; |  | ||||||
|             } |  | ||||||
|             followMsg.add_embed(followEmbed); |  | ||||||
|  |  | ||||||
|             botCluster->message_create(followMsg); |  | ||||||
|         } |         } | ||||||
|     }); |         msg.content = "지금 재생 중:"; | ||||||
|  |         msg.add_embed(nowplaying.embed); | ||||||
|  |  | ||||||
|  |         for (auto iter = queue.begin(); iter != queue.end(); iter++) { | ||||||
|  |             dpp::message followMsg(event.command.channel_id, ""); | ||||||
|  |  | ||||||
|  |             followMsg.add_embed(iter->embed); | ||||||
|  |             event.from->creator->message_create(followMsg); // 어차피 원래 메시지를 지정해서 수정할 것이기 때문에 먼저 팔로잉 메시지를 작성해도 상관없음. | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         event.edit_original_response(msg); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void Queue::init() { | ||||||
|  |     } | ||||||
| } | } | ||||||
| @@ -1,27 +1,16 @@ | |||||||
| #include <Commands/Repeat.hpp> | #include <Commands/BumbleBeeCommand.hpp> | ||||||
| #include <dpp/dpp.h> |  | ||||||
| #include <string> |  | ||||||
|  |  | ||||||
| commands::Repeat::Repeat(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap) | namespace bumbleBee::commands { | ||||||
|     : VCCommand(botCluster) |     void Repeat::execute(const dpp::slashcommand_t &event) { | ||||||
| { |         if (musicManager->getRepeat(event.command.guild_id)) { | ||||||
|     this->queueMap = queueMap; |             event.edit_original_response(dpp::message("반복을 껐습니다.")); | ||||||
|     dpp::slashcommand command = dpp::slashcommand("r", "반복 켜기/끄기", botCluster->me.id); |             musicManager->setRepeat(event.command.guild_id, false); | ||||||
|  |         } | ||||||
|     commandObjectVector.push_back(command); |         else { | ||||||
| } |             event.edit_original_response(dpp::message("반복을 켰습니다.")); | ||||||
|  |             musicManager->setRepeat(event.command.guild_id, true); | ||||||
| void commands::Repeat::operator()(const dpp::slashcommand_t& event) { |         } | ||||||
|     std::shared_ptr<MusicQueue> queue = getQueue(event); |  | ||||||
|  |  | ||||||
|     if (queue->repeat) { |  | ||||||
|         event.reply("반복을 껐습니다."); |  | ||||||
|         queue->repeat = false; |  | ||||||
|     } |  | ||||||
|     else { |  | ||||||
|         event.reply("반복을 켰습니다."); |  | ||||||
|         queue->repeat = true; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return; |     void Repeat::init() {} | ||||||
| } | } | ||||||
							
								
								
									
										10
									
								
								src/Commands/Shuffle.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/Commands/Shuffle.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | #include <Commands/BumbleBeeCommand.hpp> | ||||||
|  |  | ||||||
|  | namespace bumbleBee::commands { | ||||||
|  |     void Shuffle::execute(const dpp::slashcommand_t &event) { | ||||||
|  |         event.edit_original_response(dpp::message("shuffle")); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void Shuffle::init() { | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,29 +1,20 @@ | |||||||
| #include <Commands/Skip.hpp> | #include <Commands/BumbleBeeCommand.hpp> | ||||||
| #include <dpp/dpp.h> |  | ||||||
| #include <string> |  | ||||||
|  |  | ||||||
| commands::Skip::Skip(std::shared_ptr<dpp::cluster> botCluster, std::unordered_map<dpp::snowflake, std::shared_ptr<MusicQueue>> *queueMap) | namespace bumbleBee::commands { | ||||||
|     : VCCommand(botCluster) |     void Skip::execute(const dpp::slashcommand_t &event) { | ||||||
| { |         dpp::voiceconn* v = event.from->get_voice(event.command.guild_id); | ||||||
|     this->queueMap = queueMap; |  | ||||||
|     dpp::slashcommand command = dpp::slashcommand("s", "현재곡 스킵", botCluster->me.id); |  | ||||||
|  |  | ||||||
|     commandObjectVector.push_back(command); |         if (!v || !v->voiceclient || !v->voiceclient->is_ready()) { | ||||||
| } |             return; | ||||||
|  |         } | ||||||
|  |         v->voiceclient->pause_audio(true); | ||||||
|  |         v->voiceclient->stop_audio(); | ||||||
|  |         v->voiceclient->pause_audio(false); | ||||||
|  |         v->voiceclient->insert_marker("end"); | ||||||
|  |  | ||||||
| void commands::Skip::operator()(const dpp::slashcommand_t& event) { |         event.edit_original_response(dpp::message("스킵했습니다!")); | ||||||
|     dpp::voiceconn* v = event.from->get_voice(event.command.guild_id); |  | ||||||
|  |  | ||||||
|     if (!v || !v->voiceclient || !v->voiceclient->is_ready()) { |  | ||||||
|         return; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     v->voiceclient->stop_audio(); |     void Skip::init() { | ||||||
|     v->voiceclient->insert_marker("next marker"); |     } | ||||||
|  |  | ||||||
|     std::shared_ptr<MusicQueue> queue = getQueue(event); |  | ||||||
|  |  | ||||||
|     event.reply("스킵했습니다!"); |  | ||||||
|  |  | ||||||
|     return; |  | ||||||
| } | } | ||||||
| @@ -1 +0,0 @@ | |||||||
| OSeStDEAZJQ |  | ||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1,216 +0,0 @@ | |||||||
| #include <MusicQueue.hpp> |  | ||||||
| #include <ogg/ogg.h> |  | ||||||
| #include <oggz/oggz.h> |  | ||||||
| #include <opus/opusfile.h> |  | ||||||
| #include <thread> |  | ||||||
|  |  | ||||||
| MusicQueue::MusicQueue(FMusicQueueID id, std::shared_ptr<dpp::cluster> botCluster) |  | ||||||
| { |  | ||||||
|     this->id = id; |  | ||||||
|     repeat = false; |  | ||||||
|     this->botCluster = botCluster; |  | ||||||
|  |  | ||||||
|     botCluster->on_voice_track_marker([this, botCluster](const dpp::voice_track_marker_t &marker) |  | ||||||
|     { |  | ||||||
|         std::cout << marker.track_meta << " Marker reached.\n"; |  | ||||||
|  |  | ||||||
|         if (empty()) |  | ||||||
|         { |  | ||||||
|             std::cout << "Queue ended\n"; |  | ||||||
|             playMutex.unlock(); |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         auto music = pop(0); |  | ||||||
|         if (repeat) |  | ||||||
|         { |  | ||||||
|             (*this) += music; |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         markerCallback(); |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void MusicQueue::operator+=(FQueueElement operand) |  | ||||||
| { |  | ||||||
|     mutex.lock(); |  | ||||||
|     queue.push_back(operand); |  | ||||||
|     mutex.unlock(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| FQueueElement MusicQueue::pop(int index) |  | ||||||
| { |  | ||||||
|     mutex.lock(); |  | ||||||
|     auto iter = queue.begin(); |  | ||||||
|     std::advance(iter, index); |  | ||||||
|     auto returnValue = *iter; |  | ||||||
|     queue.erase(iter); |  | ||||||
|     mutex.unlock(); |  | ||||||
|  |  | ||||||
|     return returnValue; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| FQueueElement MusicQueue::peek(int index) |  | ||||||
| { |  | ||||||
|     mutex.lock(); |  | ||||||
|     auto iter = queue.begin(); |  | ||||||
|     std::advance(iter, index); |  | ||||||
|     auto returnValue = *iter; |  | ||||||
|     mutex.unlock(); |  | ||||||
|  |  | ||||||
|     return returnValue; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| bool MusicQueue::empty() |  | ||||||
| { |  | ||||||
|     mutex.lock(); |  | ||||||
|     bool empty = queue.empty(); |  | ||||||
|     mutex.unlock(); |  | ||||||
|      |  | ||||||
|     return empty; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void MusicQueue::clear() |  | ||||||
| { |  | ||||||
|     mutex.lock(); |  | ||||||
|     queue.clear(); |  | ||||||
|     mutex.unlock(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| std::list<struct FQueueElement>::iterator MusicQueue::begin() |  | ||||||
| { |  | ||||||
|     mutex.lock(); |  | ||||||
|     auto returnValue = queue.begin(); |  | ||||||
|     mutex.unlock(); |  | ||||||
|  |  | ||||||
|     return returnValue; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| std::list<struct FQueueElement>::iterator MusicQueue::end() |  | ||||||
| { |  | ||||||
|     mutex.lock(); |  | ||||||
|     auto returnValue = queue.end(); |  | ||||||
|     mutex.unlock(); |  | ||||||
|  |  | ||||||
|     return returnValue; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| std::size_t MusicQueue::size() |  | ||||||
| { |  | ||||||
|     mutex.lock(); |  | ||||||
|     auto returnValue = queue.size(); |  | ||||||
|     mutex.unlock(); |  | ||||||
|  |  | ||||||
|     return returnValue; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| FMusicQueueID MusicQueue::getId() |  | ||||||
| { |  | ||||||
|     return id; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void MusicQueue::markerCallback() |  | ||||||
| { |  | ||||||
|     std::cout << "Music play started\n"; |  | ||||||
|  |  | ||||||
|     dpp::discord_client* joinedShard = botCluster->get_shard(id.shard_id); |  | ||||||
|     if (!joinedShard) |  | ||||||
|     { |  | ||||||
|         std::cout << "No shard\n"; |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (empty()) |  | ||||||
|     { |  | ||||||
|         std::cout << "Queue ended\n"; |  | ||||||
|         playMutex.unlock(); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     FQueueElement music = peek(0); |  | ||||||
|  |  | ||||||
|     dpp::voiceconn* v = joinedShard->get_voice(id.guild_id); |  | ||||||
|     if (!v || !v->voiceclient || !v->voiceclient->is_ready()) |  | ||||||
|     { |  | ||||||
|         std::cout << "not in voicechat. quit musicplay"; |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     //v->voiceclient->error(4014); |  | ||||||
|  |  | ||||||
|     /* load the audio file with oggz */ |  | ||||||
|     OGGZ *track_og = oggz_open(("Music/" + music.fileName + ".ogg").c_str(), OGGZ_READ); |  | ||||||
|  |  | ||||||
|     /* If there was an issue reading the file, tell the user and stop */ |  | ||||||
|     if (!track_og) { |  | ||||||
|         fprintf(stderr, "Error opening file\n"); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /* set read callback, this callback will be called on packets with the serialno, |  | ||||||
|         * -1 means every packet will be handled with this callback. |  | ||||||
|         */ |  | ||||||
|     oggz_set_read_callback( |  | ||||||
|         track_og, -1, |  | ||||||
|         [](OGGZ *oggz, oggz_packet *packet, long serialno, void *user_data) { |  | ||||||
|             dpp::voiceconn *voiceconn = (dpp::voiceconn *)user_data; |  | ||||||
|  |  | ||||||
|             /* send the audio */ |  | ||||||
|             voiceconn->voiceclient->send_audio_opus(packet->op.packet, packet->op.bytes); |  | ||||||
|  |  | ||||||
|             /* make sure to always return 0 here, read more on oggz documentation */ |  | ||||||
|             return 0; |  | ||||||
|         }, |  | ||||||
|         /* this will be the value of void *user_data */ |  | ||||||
|         (void *)v |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     // read loop |  | ||||||
|     while (v && v->voiceclient && !v->voiceclient->terminating) { |  | ||||||
|         /* you can tweak this to whatever. Here I use BUFSIZ, defined in |  | ||||||
|             * stdio.h as 8192. |  | ||||||
|             */ |  | ||||||
|         static const constexpr long CHUNK_READ = BUFSIZ * 2; |  | ||||||
|  |  | ||||||
|         const long read_bytes = oggz_read(track_og, CHUNK_READ); |  | ||||||
|  |  | ||||||
|         /* break on eof */ |  | ||||||
|         if (!read_bytes) { |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /* Don't forget to free the memory */ |  | ||||||
|     oggz_close(track_og); |  | ||||||
|  |  | ||||||
|     v->voiceclient->insert_marker(music.fileName + " end"); |  | ||||||
|  |  | ||||||
|     std::cout << "audio sending complete\n"; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void MusicQueue::play() |  | ||||||
| { |  | ||||||
|     if (!playMutex.try_lock()) |  | ||||||
|     { |  | ||||||
|         std::cout << "Already playing\n"; |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     dpp::discord_client* joinedShard = botCluster->get_shard(id.shard_id); |  | ||||||
|     if (!joinedShard) |  | ||||||
|     { |  | ||||||
|         std::cout << "No shard\n"; |  | ||||||
|         playMutex.unlock(); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     dpp::voiceconn* v = joinedShard->get_voice(id.guild_id); |  | ||||||
|     if (!v || !v->voiceclient || !v->voiceclient->is_ready()) |  | ||||||
|     { |  | ||||||
|         std::cout << "not in voicechat. quit musicplay"; |  | ||||||
|         playMutex.unlock(); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|      |  | ||||||
|     markerCallback(); |  | ||||||
| } |  | ||||||
							
								
								
									
										97
									
								
								src/Queue/MusicQueue.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								src/Queue/MusicQueue.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | |||||||
|  | #include <Queue/MusicQueue.hpp> | ||||||
|  | #include <iostream> | ||||||
|  | #include <algorithm> | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  |  | ||||||
|  | void MusicQueue::enqueue(std::shared_ptr<MusicQueueElement> Element) { | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |         queue.push_back(Element); | ||||||
|  |     if (queue.size() == 1) // 이전에 하나도 없었다는 말이므로. | ||||||
|  |         currentPlayingPosition = queue.begin(); // 첫번째 곡으로 iterator를 옮겨준다. | ||||||
|  | } | ||||||
|  |  | ||||||
|  | std::shared_ptr<MusicQueueElement> MusicQueue::dequeue() { | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |     auto value = queue.front(); | ||||||
|  |     queue.pop_front(); | ||||||
|  |     return value; | ||||||
|  | } | ||||||
|  | std::list<std::shared_ptr<MusicQueueElement>>::iterator MusicQueue::findById(std::string id) { | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |     int index = 0; | ||||||
|  |     for (auto iter = queue.begin(); iter != queue.end(); iter++) { | ||||||
|  |         if ((*iter).get()->id == id) | ||||||
|  |             return iter; | ||||||
|  |     } | ||||||
|  |     return queue.end(); | ||||||
|  | } | ||||||
|  | std::list<std::shared_ptr<MusicQueueElement>>::iterator MusicQueue::findByIndex(int index) { | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |     if (index < 0) | ||||||
|  |         return queue.end(); | ||||||
|  |     if (index == 0) | ||||||
|  |         return currentPlayingPosition; | ||||||
|  |     return std::next(queue.begin(), index - 1); | ||||||
|  | } | ||||||
|  | std::shared_ptr<MusicQueueElement> MusicQueue::nowplaying() { | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |     return *currentPlayingPosition; | ||||||
|  | } | ||||||
|  | std::list<std::shared_ptr<MusicQueueElement>>::iterator MusicQueue::next_music() { | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |     if (currentPlayingPosition == --queue.end() && !repeat) | ||||||
|  |         return queue.end(); | ||||||
|  |     if (currentPlayingPosition == --queue.end() && repeat) | ||||||
|  |         currentPlayingPosition = queue.begin(); | ||||||
|  |     else | ||||||
|  |         ++currentPlayingPosition; | ||||||
|  |     return currentPlayingPosition; | ||||||
|  | } | ||||||
|  | std::shared_ptr<MusicQueueElement> MusicQueue::jump_to_index(int idx) { | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |     int index = 0; | ||||||
|  |     for (auto iter = queue.begin(); iter != queue.end(); iter++) { | ||||||
|  |         if (idx == index++) { | ||||||
|  |             currentPlayingPosition = iter; | ||||||
|  |             return *iter; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return std::shared_ptr<MusicQueueElement>(); | ||||||
|  | } | ||||||
|  | void MusicQueue::clear() { | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |     queue.clear(); | ||||||
|  |     currentPlayingPosition = queue.begin(); | ||||||
|  | } | ||||||
|  | std::shared_ptr<MusicQueueElement> MusicQueue::erase(std::list<std::shared_ptr<MusicQueueElement>>::iterator it) { | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |     if (it == queue.end()) | ||||||
|  |         return nullptr; | ||||||
|  |      | ||||||
|  |     if (currentPlayingPosition == it) { | ||||||
|  |         auto removedValue = *it; | ||||||
|  |         auto after = queue.erase(currentPlayingPosition++); | ||||||
|  |         if (after == queue.end()) | ||||||
|  |             currentPlayingPosition = --queue.end(); | ||||||
|  |         return removedValue; | ||||||
|  |     } | ||||||
|  |     else { | ||||||
|  |         auto removedValue = *it; | ||||||
|  |         queue.erase(it); | ||||||
|  |         return removedValue; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | std::list<std::shared_ptr<MusicQueueElement>> MusicQueue::getQueueCopy(){ | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |     std::list<std::shared_ptr<MusicQueueElement>> returnValue; | ||||||
|  |  | ||||||
|  |     std::copy(queue.begin(), queue.end(), std::back_inserter(returnValue)); | ||||||
|  |  | ||||||
|  |     return returnValue;  | ||||||
|  | } | ||||||
|  | int MusicQueue::size() { | ||||||
|  |     std::lock_guard<std::mutex> lock(queueMutex); | ||||||
|  |     return queue.size(); | ||||||
|  | } | ||||||
|  | } | ||||||
							
								
								
									
										118
									
								
								src/Settings/SettingsManager.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								src/Settings/SettingsManager.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | |||||||
|  | #include <Settings/SettingsManager.hpp> | ||||||
|  | #include <dpp/nlohmann/json.hpp> | ||||||
|  | #include <Utils/ConsoleUtils.hpp> | ||||||
|  | #include <fstream> | ||||||
|  | #include <sstream> | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  |  | ||||||
|  | std::string SettingsManager::TOKEN = ""; | ||||||
|  | std::string SettingsManager::YTDLP_CMD = "./yt-dlp"; | ||||||
|  | std::string SettingsManager::FFMPEG_CMD = "./ffmpeg/bin/ffmpeg"; | ||||||
|  | dpp::loglevel SettingsManager::LOGLEVEL = dpp::ll_debug; | ||||||
|  | bool SettingsManager::REGISTER_COMMAND = false; | ||||||
|  |  | ||||||
|  | bool SettingsManager::validateToken() { | ||||||
|  |     nlohmann::json response; | ||||||
|  |     if (ConsoleUtils::getResultFromCommand("which curl").size() == 0) { | ||||||
|  |         std::cout << "curl is unavaliable. unresolable error please install curl." << std::endl; | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     std::string stresult = ConsoleUtils::getResultFromCommand("curl -sX GET \"https://discord.com/api/v10/users/@me\" -H \"Authorization: Bot " + | ||||||
|  |         TOKEN + "\"").front(); | ||||||
|  |     std::stringstream ss(stresult); | ||||||
|  |     ss >> response; | ||||||
|  |  | ||||||
|  |     if (response.contains("message") && response["message"] == "401: Unauthorized") { | ||||||
|  |         std::cout << "Token is invalid" << std::endl; | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |     else | ||||||
|  |         return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool SettingsManager::load() { | ||||||
|  |     nlohmann::json configdocument; | ||||||
|  |     try { | ||||||
|  |         std::ifstream configfile("config.json"); | ||||||
|  |         if (!configfile) { | ||||||
|  |             saveToFile(); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         configfile >> configdocument; | ||||||
|  |  | ||||||
|  |         TOKEN = configdocument["TOKEN"]; | ||||||
|  |  | ||||||
|  |         if (!validateToken()) | ||||||
|  |             return false; | ||||||
|  |  | ||||||
|  |         YTDLP_CMD = configdocument["YTDLP_CMD"]; | ||||||
|  |         FFMPEG_CMD = configdocument["FFMPEG_CMD"]; | ||||||
|  |  | ||||||
|  |         std::string level = configdocument["LOGLEVEL"]; | ||||||
|  |          | ||||||
|  |         std::transform(level.begin(), level.end(), level.begin(), ::tolower); | ||||||
|  |         if (level == "trace") | ||||||
|  |             LOGLEVEL = dpp::ll_trace; | ||||||
|  |         else if (level == "debug") | ||||||
|  |             LOGLEVEL = dpp::ll_debug; | ||||||
|  |         else if (level == "warning") | ||||||
|  |             LOGLEVEL = dpp::ll_warning; | ||||||
|  |         else if (level == "error") | ||||||
|  |             LOGLEVEL = dpp::ll_error; | ||||||
|  |         else if (level == "critical") | ||||||
|  |             LOGLEVEL = dpp::ll_critical; | ||||||
|  |         else // 값이 병신같을때 기본값으로 ll_info 부여 | ||||||
|  |             LOGLEVEL = dpp::ll_info; | ||||||
|  |  | ||||||
|  |         REGISTER_COMMAND = configdocument["CLEAR_PREVIOUS_COMMAND"]; | ||||||
|  |     } | ||||||
|  |     catch (const nlohmann::json::type_error& e) { | ||||||
|  |         saveToFile(); | ||||||
|  |         return load(); | ||||||
|  |     } | ||||||
|  |     catch (const nlohmann::json::parse_error& e) { | ||||||
|  |         saveToFile(); | ||||||
|  |         return load(); | ||||||
|  |     } | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void SettingsManager::saveToFile() { | ||||||
|  |     nlohmann::json configdocument; | ||||||
|  |  | ||||||
|  |     configdocument["TOKEN"] = TOKEN; | ||||||
|  |     configdocument["YTDLP_CMD"] = YTDLP_CMD; | ||||||
|  |     configdocument["FFMPEG_CMD"] = FFMPEG_CMD; | ||||||
|  |  | ||||||
|  |     switch (LOGLEVEL) { | ||||||
|  |     case dpp::ll_trace: | ||||||
|  |         configdocument["LOGLEVEL"] = "trace"; | ||||||
|  |         break; | ||||||
|  |     case dpp::ll_debug: | ||||||
|  |         configdocument["LOGLEVEL"] = "debug"; | ||||||
|  |         break; | ||||||
|  |     case dpp::ll_info: | ||||||
|  |         configdocument["LOGLEVEL"] = "info"; | ||||||
|  |         break; | ||||||
|  |     case dpp::ll_warning: | ||||||
|  |         configdocument["LOGLEVEL"] = "warning"; | ||||||
|  |         break; | ||||||
|  |     case dpp::ll_error: | ||||||
|  |         configdocument["LOGLEVEL"] = "error"; | ||||||
|  |         break; | ||||||
|  |     default: | ||||||
|  |         configdocument["LOGLEVEL"] = "critical"; | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     configdocument["CLEAR_PREVIOUS_COMMAND"] = REGISTER_COMMAND; | ||||||
|  |  | ||||||
|  |     std::ofstream configFile("config.json"); | ||||||
|  |     if (configFile.is_open()) { | ||||||
|  |         configFile << configdocument.dump(4); | ||||||
|  |         configFile.close(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | } | ||||||
							
								
								
									
										86
									
								
								src/Utils/AsyncDownloadManager.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								src/Utils/AsyncDownloadManager.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,86 @@ | |||||||
|  | #include <Utils/AsyncDownloadManager.hpp> | ||||||
|  | #include <sstream> | ||||||
|  | #include "Utils/ConsoleUtils.hpp" | ||||||
|  | #include "Settings/SettingsManager.hpp" | ||||||
|  | #include <ogg/ogg.h> | ||||||
|  | #include <oggz/oggz.h> | ||||||
|  | #include <opus/opusfile.h> | ||||||
|  | #include <memory> | ||||||
|  |  | ||||||
|  | namespace bumbleBee { | ||||||
|  | void AsyncDownloadManager::enqueueAsyncDL(std::pair<std::string, dpp::message> query) { | ||||||
|  |     std::lock_guard<std::mutex> lock(dlQueueMutex); | ||||||
|  |     downloadQueue.push(query); | ||||||
|  |     dlQueueCondition.notify_one(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void AsyncDownloadManager::downloadWorker() { | ||||||
|  |     std::ostringstream tid; | ||||||
|  |     tid << std::this_thread::get_id(); | ||||||
|  |     while (true) { | ||||||
|  |         //mutex lock | ||||||
|  |         std::unique_lock<std::mutex> dlQueueLock(dlQueueMutex); | ||||||
|  |         dlQueueCondition.wait(dlQueueLock, [&]{ return !downloadQueue.empty() || terminate; }); | ||||||
|  |         auto cluster = weak_cluster.lock(); | ||||||
|  |         if (weak_cluster.expired()) { | ||||||
|  |             cluster->log(dpp::ll_error, "Missing cluster, terminating thread " + tid.str()); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         if (terminate) { | ||||||
|  |             cluster->log(dpp::ll_info, "Terminating thread " + tid.str()); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         std::string query = downloadQueue.front().first; | ||||||
|  |         dpp::message oRes = downloadQueue.front().second; | ||||||
|  |         downloadQueue.pop(); | ||||||
|  |         dlQueueLock.unlock(); | ||||||
|  |  | ||||||
|  |         cluster->log(dpp::ll_info, "AsyncDownloadManager: " + query + " accepted."); | ||||||
|  |  | ||||||
|  |         std::queue<std::string> ids = | ||||||
|  |             ConsoleUtils::getResultFromCommand(SettingsManager::getYTDLP_CMD() + | ||||||
|  |             " --default-search ytsearch --flat-playlist --skip-download --quiet --ignore-errors --print id " + query); | ||||||
|  |  | ||||||
|  |         if (ids.size() >= 2) { | ||||||
|  |             cluster->log(dpp::ll_info, query + " is playlist"); | ||||||
|  |             while (!ids.empty()) { | ||||||
|  |                 if (ids.front() == "") { | ||||||
|  |                     ids.pop(); | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  |                 cluster->log(dpp::ll_info, "Enqueuing playlist element " + ids.front()); | ||||||
|  |                 enqueue(std::make_pair("https://youtu.be/" + ids.front(), oRes)); | ||||||
|  |                 ids.pop(); | ||||||
|  |             } | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         std::queue<std::string> urls = | ||||||
|  |             ConsoleUtils::getResultFromCommand(SettingsManager::getYTDLP_CMD() + | ||||||
|  |             " -f ba* --print urls https://youtu.be/" + ids.front()); | ||||||
|  |  | ||||||
|  |         cluster->log(dpp::ll_debug, "url: " + urls.front()); | ||||||
|  |  | ||||||
|  |         FILE* stream; | ||||||
|  |  | ||||||
|  |         // musicQueue->enqueue(std::make_shared<MusicQueueElement>(oRes, ids.front(), urls.front(), stream)); | ||||||
|  |  | ||||||
|  |         std::string downloadID = ids.front(); | ||||||
|  |  | ||||||
|  |         std::thread th([&, downloadID](){ | ||||||
|  |             if (terminate) | ||||||
|  |                 return; | ||||||
|  |             std::ostringstream tid; | ||||||
|  |             tid << std::this_thread::get_id(); | ||||||
|  |  | ||||||
|  |             cluster->log(dpp::ll_info, "Thread id: " + tid.str() + ": " + downloadID + " accepted."); | ||||||
|  |  | ||||||
|  |             std::string command = std::string("./streamOpus.sh " + SettingsManager::getYTDLP_CMD() + " " + downloadID + " " + SettingsManager::getFFMPEG_CMD()); | ||||||
|  |             stream = popen(command.c_str(), "r"); | ||||||
|  |  | ||||||
|  |             cluster->log(dpp::ll_info, "Thread id: " + tid.str() + " Opened stream: " + downloadID); | ||||||
|  |         }); | ||||||
|  |         th.detach(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								src/main.cpp
									
									
									
									
									
								
							
							
						
						
									
										22
									
								
								src/main.cpp
									
									
									
									
									
								
							| @@ -1,18 +1,8 @@ | |||||||
| #include <BumbleCeepp.hpp> | #include <iostream> | ||||||
| #include <dpp/nlohmann/json.hpp> | #include <BumbleBee.hpp> | ||||||
| #include <memory> | #include <thread> | ||||||
|  |  | ||||||
| using json = nlohmann::json; | int main() { | ||||||
|  |     bumbleBee::BumbleBee bot; | ||||||
| int main() |     bot.start(); | ||||||
| { |  | ||||||
|     json configdocument; |  | ||||||
|     std::ifstream configfile("config.json"); |  | ||||||
|     configfile >> configdocument; |  | ||||||
|  |  | ||||||
|     std::shared_ptr<BumbleCeepp> bumbleBee = std::make_shared<BumbleCeepp>(configdocument["token"], 1); |  | ||||||
|  |  | ||||||
|     bumbleBee->start(); |  | ||||||
|  |  | ||||||
|     return 0; |  | ||||||
| } | } | ||||||
							
								
								
									
										
											BIN
										
									
								
								src/yt-dlp
									
									
									
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/yt-dlp
									
									
									
									
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1,47 +0,0 @@ | |||||||
| import yt_dlp |  | ||||||
| import sys |  | ||||||
| import os |  | ||||||
| import json |  | ||||||
|  |  | ||||||
| if len(sys.argv) != 2: |  | ||||||
|     sys.exit() |  | ||||||
|  |  | ||||||
| ydl_opts = { |  | ||||||
|     'quiet': True, |  | ||||||
|     'clean_infojson': False, |  | ||||||
|     'default_search': 'ytsearch', |  | ||||||
|     'format': '251', |  | ||||||
|     'outtmpl': {'default': 'Temp/%(id)s.temp'}, |  | ||||||
|     'overwrites': False, |  | ||||||
|     'writeinfojson': True } |  | ||||||
|  |  | ||||||
| with yt_dlp.YoutubeDL(ydl_opts) as ydl: |  | ||||||
|     info = ydl.extract_info(sys.argv[1], download=False) |  | ||||||
|     id = list() |  | ||||||
|     with open("out", "w") as f: |  | ||||||
|         f.write(json.dumps(ydl.sanitize_info(info))) |  | ||||||
|     with open("Music/Archive", "r") as f: |  | ||||||
|         ArchiveList = f.read().split("\n") |  | ||||||
|         if "entries" in info: |  | ||||||
|             if len(info["entries"]) != 0: |  | ||||||
|                 for entry in info["entries"]: |  | ||||||
|                     if entry["id"] not in ArchiveList: |  | ||||||
|                         ydl.download(entry["webpage_url"]) |  | ||||||
|                         os.system("echo " + entry["id"] + " >> Music/Archive") |  | ||||||
|                         os.system("yes n 2>/dev/null | ffmpeg -hide_banner -loglevel error -i \"" + "Temp/" + entry["id"] + ".temp" + "\" -c copy Music/" + entry["id"] + ".ogg") |  | ||||||
|                         os.system("mv Temp/" + entry["id"] + ".temp.info.json Music/" + entry["id"] + ".info.json") |  | ||||||
|                     id.append(entry["id"]) |  | ||||||
|         else: |  | ||||||
|             if info["id"] not in ArchiveList: |  | ||||||
|                 ydl.download(info["webpage_url"]) |  | ||||||
|                 os.system("echo " + info["id"] + " >> Music/Archive") |  | ||||||
|                 os.system("yes n 2>/dev/null | ffmpeg -hide_banner -loglevel error -i \"" + "Temp/" + info["id"] + ".temp" + "\" -c copy Music/" + info["id"] + ".ogg") |  | ||||||
|                 os.system("mv Temp/" + info["id"] + ".temp.info.json Music/" + info["id"] + ".info.json") |  | ||||||
|             id.append(info["id"]) |  | ||||||
|  |  | ||||||
|     os.system("rm -f Temp/*.temp") |  | ||||||
|     os.system("rm -f Temp/*.json") |  | ||||||
|  |  | ||||||
|     with open("Temp/CurMusic", "w") as f: |  | ||||||
|         for item in id: |  | ||||||
|             f.write(item + "\n") |  | ||||||
							
								
								
									
										12
									
								
								streamOpus.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										12
									
								
								streamOpus.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | YTDLP_CMD=$1 | ||||||
|  | FFMPEG_CMD=$2 | ||||||
|  | URL=$3 | ||||||
|  |  | ||||||
|  | $YTDLP_CMD -o - --quiet --ignore-errors -f bestaudio $URL 2>/dev/null | \ | ||||||
|  | $FFMPEG_CMD -hwaccel vaapi -i - -loglevel quiet -hide_banner -c:a copy -f opus - || \ | ||||||
|  | $YTDLP_CMD -o - --quiet --ignore-errors -f bestaudio $URL 2>/dev/null | \ | ||||||
|  | $FFMPEG_CMD -hwaccel vaapi -i - -loglevel quiet -hide_banner -f opus -c:a libopus -b:a 128k -ar 48000 -ac 2 - | ||||||
|  |  | ||||||
|  | #  -loglevel quiet | ||||||
		Reference in New Issue
	
	Block a user