diff options
author | UbitUmarov | 2015-09-01 11:43:07 +0100 |
---|---|---|
committer | UbitUmarov | 2015-09-01 11:43:07 +0100 |
commit | fb78b182520fc9bb0f971afd0322029c70278ea6 (patch) | |
tree | b4e30d383938fdeef8c92d1d1c2f44bb61d329bd /OpenSim/Tools | |
parent | lixo (diff) | |
parent | Mantis #7713: fixed bug introduced by 1st MOSES patch. (diff) | |
download | opensim-SC-fb78b182520fc9bb0f971afd0322029c70278ea6.zip opensim-SC-fb78b182520fc9bb0f971afd0322029c70278ea6.tar.gz opensim-SC-fb78b182520fc9bb0f971afd0322029c70278ea6.tar.bz2 opensim-SC-fb78b182520fc9bb0f971afd0322029c70278ea6.tar.xz |
Merge remote-tracking branch 'os/master'
Diffstat (limited to 'OpenSim/Tools')
43 files changed, 5195 insertions, 0 deletions
diff --git a/OpenSim/Tools/Compiler/Program.cs b/OpenSim/Tools/Compiler/Program.cs new file mode 100644 index 0000000..b010eaf --- /dev/null +++ b/OpenSim/Tools/Compiler/Program.cs | |||
@@ -0,0 +1,325 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.Collections.Generic; | ||
30 | using System.IO; | ||
31 | using System.Text; | ||
32 | using Microsoft.CSharp; | ||
33 | using OpenSim.Region.ScriptEngine.Shared.CodeTools; | ||
34 | using System.CodeDom.Compiler; | ||
35 | |||
36 | namespace OpenSim.Tools.LSL.Compiler | ||
37 | { | ||
38 | class Program | ||
39 | { | ||
40 | // Commented out because generated warning since m_positionMap could never be anything other than null | ||
41 | // private static Dictionary<KeyValuePair<int, int>, KeyValuePair<int, int>> m_positionMap; | ||
42 | private static CSharpCodeProvider CScodeProvider = new CSharpCodeProvider(); | ||
43 | |||
44 | static void Main(string[] args) | ||
45 | { | ||
46 | string source = null; | ||
47 | |||
48 | if (args.Length == 0) | ||
49 | { | ||
50 | Console.WriteLine("No input file specified"); | ||
51 | Environment.Exit(1); | ||
52 | } | ||
53 | |||
54 | if (!File.Exists(args[0])) | ||
55 | { | ||
56 | Console.WriteLine("Input file does not exist"); | ||
57 | Environment.Exit(1); | ||
58 | } | ||
59 | |||
60 | try | ||
61 | { | ||
62 | ICodeConverter cvt = (ICodeConverter) new CSCodeGenerator(); | ||
63 | source = cvt.Convert(File.ReadAllText(args[0])); | ||
64 | } | ||
65 | catch(Exception e) | ||
66 | { | ||
67 | Console.WriteLine("Conversion failed:\n"+e.Message); | ||
68 | Environment.Exit(1); | ||
69 | } | ||
70 | |||
71 | source = CreateCSCompilerScript(source); | ||
72 | |||
73 | try | ||
74 | { | ||
75 | Console.WriteLine(CompileFromDotNetText(source,"a.out")); | ||
76 | } | ||
77 | catch(Exception e) | ||
78 | { | ||
79 | Console.WriteLine("Conversion failed: "+e.Message); | ||
80 | Environment.Exit(1); | ||
81 | } | ||
82 | |||
83 | Environment.Exit(0); | ||
84 | } | ||
85 | |||
86 | private static string CreateCSCompilerScript(string compileScript) | ||
87 | { | ||
88 | compileScript = String.Empty + | ||
89 | "using OpenSim.Region.ScriptEngine.Shared; using System.Collections.Generic;\r\n" + | ||
90 | String.Empty + "namespace SecondLife { " + | ||
91 | String.Empty + "public class Script : OpenSim.Region.ScriptEngine.Shared.ScriptBase.ScriptBaseClass { \r\n" + | ||
92 | @"public Script() { } " + | ||
93 | compileScript + | ||
94 | "} }\r\n"; | ||
95 | return compileScript; | ||
96 | } | ||
97 | |||
98 | private static string CompileFromDotNetText(string Script, string asset) | ||
99 | { | ||
100 | |||
101 | string OutFile = asset; | ||
102 | string disp ="OK"; | ||
103 | |||
104 | try | ||
105 | { | ||
106 | File.Delete(OutFile); | ||
107 | } | ||
108 | catch (Exception e) // NOTLEGIT - Should be just FileIOException | ||
109 | { | ||
110 | throw new Exception("Unable to delete old existing "+ | ||
111 | "script-file before writing new. Compile aborted: " + | ||
112 | e.ToString()); | ||
113 | } | ||
114 | |||
115 | // Do actual compile | ||
116 | CompilerParameters parameters = new CompilerParameters(); | ||
117 | |||
118 | parameters.IncludeDebugInformation = true; | ||
119 | |||
120 | string rootPath = | ||
121 | Path.GetDirectoryName(AppDomain.CurrentDomain.BaseDirectory); | ||
122 | |||
123 | parameters.ReferencedAssemblies.Add(Path.Combine(rootPath, | ||
124 | "OpenSim.Region.ScriptEngine.Shared.dll")); | ||
125 | parameters.ReferencedAssemblies.Add(Path.Combine(rootPath, | ||
126 | "OpenSim.Region.ScriptEngine.Shared.Api.Runtime.dll")); | ||
127 | |||
128 | parameters.GenerateExecutable = false; | ||
129 | parameters.OutputAssembly = OutFile; | ||
130 | parameters.IncludeDebugInformation = true; | ||
131 | parameters.WarningLevel = 1; | ||
132 | parameters.TreatWarningsAsErrors = false; | ||
133 | |||
134 | CompilerResults results = CScodeProvider.CompileAssemblyFromSource(parameters, Script); | ||
135 | |||
136 | if (results.Errors.Count > 0) | ||
137 | { | ||
138 | string errtext = String.Empty; | ||
139 | foreach (CompilerError CompErr in results.Errors) | ||
140 | { | ||
141 | string severity = CompErr.IsWarning ? "Warning" : "Error"; | ||
142 | |||
143 | KeyValuePair<int, int> lslPos; | ||
144 | |||
145 | lslPos = FindErrorPosition(CompErr.Line, CompErr.Column); | ||
146 | |||
147 | string text = CompErr.ErrorText; | ||
148 | |||
149 | text = ReplaceTypes(CompErr.ErrorText); | ||
150 | |||
151 | // The Second Life viewer's script editor begins | ||
152 | // countingn lines and columns at 0, so we subtract 1. | ||
153 | errtext += String.Format("Line ({0},{1}): {4} {2}: {3}\n", | ||
154 | lslPos.Key - 1, lslPos.Value - 1, | ||
155 | CompErr.ErrorNumber, text, severity); | ||
156 | } | ||
157 | |||
158 | disp = "Completed with errors"; | ||
159 | |||
160 | if (!File.Exists(OutFile)) | ||
161 | { | ||
162 | throw new Exception(errtext); | ||
163 | } | ||
164 | } | ||
165 | |||
166 | if (!File.Exists(OutFile)) | ||
167 | { | ||
168 | string errtext = String.Empty; | ||
169 | errtext += "No compile error. But not able to locate compiled file."; | ||
170 | throw new Exception(errtext); | ||
171 | } | ||
172 | |||
173 | // Because windows likes to perform exclusive locks, we simply | ||
174 | // write out a textual representation of the file here | ||
175 | // | ||
176 | // Read the binary file into a buffer | ||
177 | // | ||
178 | FileInfo fi = new FileInfo(OutFile); | ||
179 | |||
180 | if (fi == null) | ||
181 | { | ||
182 | string errtext = String.Empty; | ||
183 | errtext += "No compile error. But not able to stat file."; | ||
184 | throw new Exception(errtext); | ||
185 | } | ||
186 | |||
187 | Byte[] data = new Byte[fi.Length]; | ||
188 | |||
189 | try | ||
190 | { | ||
191 | FileStream fs = File.Open(OutFile, FileMode.Open, FileAccess.Read); | ||
192 | fs.Read(data, 0, data.Length); | ||
193 | fs.Close(); | ||
194 | } | ||
195 | catch (Exception) | ||
196 | { | ||
197 | string errtext = String.Empty; | ||
198 | errtext += "No compile error. But not able to open file."; | ||
199 | throw new Exception(errtext); | ||
200 | } | ||
201 | |||
202 | // Convert to base64 | ||
203 | // | ||
204 | string filetext = System.Convert.ToBase64String(data); | ||
205 | Byte[] buf = Encoding.ASCII.GetBytes(filetext); | ||
206 | FileStream sfs = File.Create(OutFile + ".text"); | ||
207 | sfs.Write(buf, 0, buf.Length); | ||
208 | sfs.Close(); | ||
209 | |||
210 | string posmap = String.Empty; | ||
211 | // if (m_positionMap != null) | ||
212 | // { | ||
213 | // foreach (KeyValuePair<KeyValuePair<int, int>, KeyValuePair<int, int>> kvp in m_positionMap) | ||
214 | // { | ||
215 | // KeyValuePair<int, int> k = kvp.Key; | ||
216 | // KeyValuePair<int, int> v = kvp.Value; | ||
217 | // posmap += String.Format("{0},{1},{2},{3}\n", | ||
218 | // k.Key, k.Value, v.Key, v.Value); | ||
219 | // } | ||
220 | // } | ||
221 | |||
222 | buf = Encoding.ASCII.GetBytes(posmap); | ||
223 | |||
224 | FileStream mfs = File.Create(OutFile + ".map"); | ||
225 | mfs.Write(buf, 0, buf.Length); | ||
226 | mfs.Close(); | ||
227 | |||
228 | return disp; | ||
229 | } | ||
230 | |||
231 | private static string ReplaceTypes(string message) | ||
232 | { | ||
233 | message = message.Replace( | ||
234 | "OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLString", | ||
235 | "string"); | ||
236 | |||
237 | message = message.Replace( | ||
238 | "OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLInteger", | ||
239 | "integer"); | ||
240 | |||
241 | message = message.Replace( | ||
242 | "OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLFloat", | ||
243 | "float"); | ||
244 | |||
245 | message = message.Replace( | ||
246 | "OpenSim.Region.ScriptEngine.Shared.LSL_Types.list", | ||
247 | "list"); | ||
248 | |||
249 | return message; | ||
250 | } | ||
251 | |||
252 | private static KeyValuePair<int, int> FindErrorPosition(int line, int col) | ||
253 | { | ||
254 | //return FindErrorPosition(line, col, m_positionMap); | ||
255 | return FindErrorPosition(line, col, null); | ||
256 | } | ||
257 | |||
258 | private class kvpSorter : IComparer<KeyValuePair<KeyValuePair<int, int>, KeyValuePair<int, int>>> | ||
259 | { | ||
260 | public int Compare(KeyValuePair<KeyValuePair<int, int>, KeyValuePair<int, int>> a, | ||
261 | KeyValuePair<KeyValuePair<int, int>, KeyValuePair<int, int>> b) | ||
262 | { | ||
263 | int kc = a.Key.Key.CompareTo(b.Key.Key); | ||
264 | return (kc != 0) ? kc : a.Key.Value.CompareTo(b.Key.Value); | ||
265 | } | ||
266 | } | ||
267 | |||
268 | public static KeyValuePair<int, int> FindErrorPosition(int line, | ||
269 | int col, Dictionary<KeyValuePair<int, int>, | ||
270 | KeyValuePair<int, int>> positionMap) | ||
271 | { | ||
272 | if (positionMap == null || positionMap.Count == 0) | ||
273 | return new KeyValuePair<int, int>(line, col); | ||
274 | |||
275 | KeyValuePair<int, int> ret = new KeyValuePair<int, int>(); | ||
276 | |||
277 | if (positionMap.TryGetValue(new KeyValuePair<int, int>(line, col), | ||
278 | out ret)) | ||
279 | return ret; | ||
280 | |||
281 | var sorted = new List<KeyValuePair<KeyValuePair<int, int>, KeyValuePair<int, int>>>(positionMap); | ||
282 | |||
283 | sorted.Sort(new kvpSorter()); | ||
284 | |||
285 | int l = 1; | ||
286 | int c = 1; | ||
287 | int pl = 1; | ||
288 | |||
289 | foreach (KeyValuePair<KeyValuePair<int, int>, KeyValuePair<int, int>> posmap in sorted) | ||
290 | { | ||
291 | //m_log.DebugFormat("[Compiler]: Scanning line map {0},{1} --> {2},{3}", posmap.Key.Key, posmap.Key.Value, posmap.Value.Key, posmap.Value.Value); | ||
292 | int nl = posmap.Value.Key + line - posmap.Key.Key; // New, translated LSL line and column. | ||
293 | int nc = posmap.Value.Value + col - posmap.Key.Value; | ||
294 | // Keep going until we find the first point passed line,col. | ||
295 | if (posmap.Key.Key > line) | ||
296 | { | ||
297 | //m_log.DebugFormat("[Compiler]: Line is larger than requested {0},{1}, returning {2},{3}", line, col, l, c); | ||
298 | if (pl < line) | ||
299 | { | ||
300 | //m_log.DebugFormat("[Compiler]: Previous line ({0}) is less than requested line ({1}), setting column to 1.", pl, line); | ||
301 | c = 1; | ||
302 | } | ||
303 | break; | ||
304 | } | ||
305 | if (posmap.Key.Key == line && posmap.Key.Value > col) | ||
306 | { | ||
307 | // Never move l,c backwards. | ||
308 | if (nl > l || (nl == l && nc > c)) | ||
309 | { | ||
310 | //m_log.DebugFormat("[Compiler]: Using offset relative to this: {0} + {1} - {2}, {3} + {4} - {5} = {6}, {7}", | ||
311 | // posmap.Value.Key, line, posmap.Key.Key, posmap.Value.Value, col, posmap.Key.Value, nl, nc); | ||
312 | l = nl; | ||
313 | c = nc; | ||
314 | } | ||
315 | //m_log.DebugFormat("[Compiler]: Column is larger than requested {0},{1}, returning {2},{3}", line, col, l, c); | ||
316 | break; | ||
317 | } | ||
318 | pl = posmap.Key.Key; | ||
319 | l = posmap.Value.Key; | ||
320 | c = posmap.Value.Value; | ||
321 | } | ||
322 | return new KeyValuePair<int, int>(l, c); | ||
323 | } | ||
324 | } | ||
325 | } | ||
diff --git a/OpenSim/Tools/Compiler/Properties/AssemblyInfo.cs b/OpenSim/Tools/Compiler/Properties/AssemblyInfo.cs new file mode 100644 index 0000000..b97c977 --- /dev/null +++ b/OpenSim/Tools/Compiler/Properties/AssemblyInfo.cs | |||
@@ -0,0 +1,33 @@ | |||
1 | using System.Reflection; | ||
2 | using System.Runtime.CompilerServices; | ||
3 | using System.Runtime.InteropServices; | ||
4 | |||
5 | // General Information about an assembly is controlled through the following | ||
6 | // set of attributes. Change these attribute values to modify the information | ||
7 | // associated with an assembly. | ||
8 | [assembly: AssemblyTitle("OpenSim.Tools.lslc")] | ||
9 | [assembly: AssemblyDescription("")] | ||
10 | [assembly: AssemblyConfiguration("")] | ||
11 | [assembly: AssemblyCompany("http://opensimulator.org")] | ||
12 | [assembly: AssemblyProduct("OpenSim")] | ||
13 | [assembly: AssemblyCopyright("OpenSimulator developers")] | ||
14 | [assembly: AssemblyTrademark("")] | ||
15 | [assembly: AssemblyCulture("")] | ||
16 | |||
17 | // Setting ComVisible to false makes the types in this assembly not visible | ||
18 | // to COM components. If you need to access a type in this assembly from | ||
19 | // COM, set the ComVisible attribute to true on that type. | ||
20 | [assembly: ComVisible(false)] | ||
21 | |||
22 | // The following GUID is for the ID of the typelib if this project is exposed to COM | ||
23 | [assembly: Guid("002864e7-b2a2-41d2-add8-82f653663160")] | ||
24 | |||
25 | // Version information for an assembly consists of the following four values: | ||
26 | // | ||
27 | // Major Version | ||
28 | // Minor Version | ||
29 | // Build Number | ||
30 | // Revision | ||
31 | // | ||
32 | [assembly: AssemblyVersion("0.8.2.*")] | ||
33 | |||
diff --git a/OpenSim/Tools/Configger/ConfigurationLoader.cs b/OpenSim/Tools/Configger/ConfigurationLoader.cs new file mode 100644 index 0000000..f1d3649 --- /dev/null +++ b/OpenSim/Tools/Configger/ConfigurationLoader.cs | |||
@@ -0,0 +1,254 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.Collections.Generic; | ||
30 | using System.IO; | ||
31 | using System.Reflection; | ||
32 | using System.Threading; | ||
33 | using System.Xml; | ||
34 | using log4net; | ||
35 | using Nini.Config; | ||
36 | |||
37 | namespace OpenSim.Tools.Configger | ||
38 | { | ||
39 | /// <summary> | ||
40 | /// Loads the Configuration files into nIni | ||
41 | /// </summary> | ||
42 | public class ConfigurationLoader | ||
43 | { | ||
44 | /// <summary> | ||
45 | /// A source of Configuration data | ||
46 | /// </summary> | ||
47 | protected IConfigSource m_config; | ||
48 | |||
49 | /// <summary> | ||
50 | /// Console logger | ||
51 | /// </summary> | ||
52 | private static readonly ILog m_log = | ||
53 | LogManager.GetLogger( | ||
54 | MethodBase.GetCurrentMethod().DeclaringType); | ||
55 | |||
56 | public ConfigurationLoader() | ||
57 | { | ||
58 | } | ||
59 | |||
60 | /// <summary> | ||
61 | /// Loads the region configuration | ||
62 | /// </summary> | ||
63 | /// <param name="argvSource">Parameters passed into the process when started</param> | ||
64 | /// <param name="configSettings"></param> | ||
65 | /// <param name="networkInfo"></param> | ||
66 | /// <returns>A configuration that gets passed to modules</returns> | ||
67 | public IConfigSource LoadConfigSettings(IConfig startupConfig) | ||
68 | { | ||
69 | bool iniFileExists = false; | ||
70 | |||
71 | List<string> sources = new List<string>(); | ||
72 | |||
73 | string iniFileName = startupConfig.GetString("inifile", Path.Combine(".", "OpenSim.ini")); | ||
74 | |||
75 | if (IsUri(iniFileName)) | ||
76 | { | ||
77 | if (!sources.Contains(iniFileName)) | ||
78 | sources.Add(iniFileName); | ||
79 | } | ||
80 | else | ||
81 | { | ||
82 | if (File.Exists(iniFileName)) | ||
83 | { | ||
84 | if (!sources.Contains(iniFileName)) | ||
85 | sources.Add(iniFileName); | ||
86 | } | ||
87 | } | ||
88 | |||
89 | m_config = new IniConfigSource(); | ||
90 | m_config.Merge(DefaultConfig()); | ||
91 | |||
92 | m_log.Info("[CONFIG] Reading configuration settings"); | ||
93 | |||
94 | if (sources.Count == 0) | ||
95 | { | ||
96 | m_log.FatalFormat("[CONFIG] Could not load any configuration"); | ||
97 | m_log.FatalFormat("[CONFIG] Did you copy the OpenSim.ini.example file to OpenSim.ini?"); | ||
98 | Environment.Exit(1); | ||
99 | } | ||
100 | |||
101 | for (int i = 0 ; i < sources.Count ; i++) | ||
102 | { | ||
103 | if (ReadConfig(sources[i])) | ||
104 | iniFileExists = true; | ||
105 | AddIncludes(sources); | ||
106 | } | ||
107 | |||
108 | if (!iniFileExists) | ||
109 | { | ||
110 | m_log.FatalFormat("[CONFIG] Could not load any configuration"); | ||
111 | m_log.FatalFormat("[CONFIG] Configuration exists, but there was an error loading it!"); | ||
112 | Environment.Exit(1); | ||
113 | } | ||
114 | |||
115 | return m_config; | ||
116 | } | ||
117 | |||
118 | /// <summary> | ||
119 | /// Adds the included files as ini configuration files | ||
120 | /// </summary> | ||
121 | /// <param name="sources">List of URL strings or filename strings</param> | ||
122 | private void AddIncludes(List<string> sources) | ||
123 | { | ||
124 | //loop over config sources | ||
125 | foreach (IConfig config in m_config.Configs) | ||
126 | { | ||
127 | // Look for Include-* in the key name | ||
128 | string[] keys = config.GetKeys(); | ||
129 | foreach (string k in keys) | ||
130 | { | ||
131 | if (k.StartsWith("Include-")) | ||
132 | { | ||
133 | // read the config file to be included. | ||
134 | string file = config.GetString(k); | ||
135 | if (IsUri(file)) | ||
136 | { | ||
137 | if (!sources.Contains(file)) | ||
138 | sources.Add(file); | ||
139 | } | ||
140 | else | ||
141 | { | ||
142 | string basepath = Path.GetFullPath("."); | ||
143 | // Resolve relative paths with wildcards | ||
144 | string chunkWithoutWildcards = file; | ||
145 | string chunkWithWildcards = string.Empty; | ||
146 | int wildcardIndex = file.IndexOfAny(new char[] { '*', '?' }); | ||
147 | if (wildcardIndex != -1) | ||
148 | { | ||
149 | chunkWithoutWildcards = file.Substring(0, wildcardIndex); | ||
150 | chunkWithWildcards = file.Substring(wildcardIndex); | ||
151 | } | ||
152 | string path = Path.Combine(basepath, chunkWithoutWildcards); | ||
153 | path = Path.GetFullPath(path) + chunkWithWildcards; | ||
154 | string[] paths = Util.Glob(path); | ||
155 | foreach (string p in paths) | ||
156 | { | ||
157 | if (!sources.Contains(p)) | ||
158 | sources.Add(p); | ||
159 | } | ||
160 | } | ||
161 | } | ||
162 | } | ||
163 | } | ||
164 | } | ||
165 | /// <summary> | ||
166 | /// Check if we can convert the string to a URI | ||
167 | /// </summary> | ||
168 | /// <param name="file">String uri to the remote resource</param> | ||
169 | /// <returns>true if we can convert the string to a Uri object</returns> | ||
170 | bool IsUri(string file) | ||
171 | { | ||
172 | Uri configUri; | ||
173 | |||
174 | return Uri.TryCreate(file, UriKind.Absolute, | ||
175 | out configUri) && configUri.Scheme == Uri.UriSchemeHttp; | ||
176 | } | ||
177 | |||
178 | /// <summary> | ||
179 | /// Provide same ini loader functionality for standard ini and master ini - file system or XML over http | ||
180 | /// </summary> | ||
181 | /// <param name="iniPath">Full path to the ini</param> | ||
182 | /// <returns></returns> | ||
183 | private bool ReadConfig(string iniPath) | ||
184 | { | ||
185 | bool success = false; | ||
186 | |||
187 | if (!IsUri(iniPath)) | ||
188 | { | ||
189 | m_log.InfoFormat("[CONFIG] Reading configuration file {0}", | ||
190 | Path.GetFullPath(iniPath)); | ||
191 | |||
192 | m_config.Merge(new IniConfigSource(iniPath)); | ||
193 | success = true; | ||
194 | } | ||
195 | else | ||
196 | { | ||
197 | m_log.InfoFormat("[CONFIG] {0} is a http:// URI, fetching ...", | ||
198 | iniPath); | ||
199 | |||
200 | // The ini file path is a http URI | ||
201 | // Try to read it | ||
202 | // | ||
203 | try | ||
204 | { | ||
205 | XmlReader r = XmlReader.Create(iniPath); | ||
206 | XmlConfigSource cs = new XmlConfigSource(r); | ||
207 | m_config.Merge(cs); | ||
208 | |||
209 | success = true; | ||
210 | } | ||
211 | catch (Exception e) | ||
212 | { | ||
213 | m_log.FatalFormat("[CONFIG] Exception reading config from URI {0}\n" + e.ToString(), iniPath); | ||
214 | Environment.Exit(1); | ||
215 | } | ||
216 | } | ||
217 | return success; | ||
218 | } | ||
219 | |||
220 | /// <summary> | ||
221 | /// Setup a default config values in case they aren't present in the ini file | ||
222 | /// </summary> | ||
223 | /// <returns>A Configuration source containing the default configuration</returns> | ||
224 | private static IConfigSource DefaultConfig() | ||
225 | { | ||
226 | IConfigSource defaultConfig = new IniConfigSource(); | ||
227 | |||
228 | { | ||
229 | IConfig config = defaultConfig.Configs["Startup"]; | ||
230 | |||
231 | if (null == config) | ||
232 | config = defaultConfig.AddConfig("Startup"); | ||
233 | |||
234 | config.Set("region_info_source", "filesystem"); | ||
235 | config.Set("allow_regionless", false); | ||
236 | |||
237 | config.Set("gridmode", false); | ||
238 | config.Set("physics", "OpenDynamicsEngine"); | ||
239 | config.Set("meshing", "Meshmerizer"); | ||
240 | config.Set("physical_prim", true); | ||
241 | config.Set("serverside_object_permissions", true); | ||
242 | config.Set("storage_prim_inventories", true); | ||
243 | config.Set("startup_console_commands_file", String.Empty); | ||
244 | config.Set("shutdown_console_commands_file", String.Empty); | ||
245 | config.Set("DefaultScriptEngine", "XEngine"); | ||
246 | config.Set("clientstack_plugin", "OpenSim.Region.ClientStack.LindenUDP.dll"); | ||
247 | // life doesn't really work without this | ||
248 | config.Set("EventQueue", true); | ||
249 | } | ||
250 | |||
251 | return defaultConfig; | ||
252 | } | ||
253 | } | ||
254 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/Configger/Main.cs b/OpenSim/Tools/Configger/Main.cs new file mode 100644 index 0000000..d7d918b --- /dev/null +++ b/OpenSim/Tools/Configger/Main.cs | |||
@@ -0,0 +1,136 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using Nini.Config; | ||
29 | using System; | ||
30 | |||
31 | namespace OpenSim.Tools.Configger | ||
32 | { | ||
33 | public class Configger | ||
34 | { | ||
35 | public static int Main(string[] args) | ||
36 | { | ||
37 | ArgvConfigSource argvConfig = new ArgvConfigSource(args); | ||
38 | |||
39 | argvConfig.AddSwitch("Startup", "format", "f"); | ||
40 | argvConfig.AddSwitch("Startup", "inifile"); | ||
41 | |||
42 | IConfig startupConfig = argvConfig.Configs["Startup"]; | ||
43 | |||
44 | string format = startupConfig.GetString("format", "ini"); | ||
45 | |||
46 | ConfigurationLoader loader = new ConfigurationLoader(); | ||
47 | IConfigSource s = loader.LoadConfigSettings(startupConfig); | ||
48 | |||
49 | if (format == "mysql") | ||
50 | { | ||
51 | foreach (IConfig c in s.Configs) | ||
52 | { | ||
53 | foreach (string k in c.GetKeys()) | ||
54 | { | ||
55 | string v = c.GetString(k); | ||
56 | |||
57 | if (k.StartsWith("Include-")) | ||
58 | continue; | ||
59 | Console.WriteLine("insert ignore into config (section, name, value) values ('{0}', '{1}', '{2}');", c.Name, k, v); | ||
60 | } | ||
61 | } | ||
62 | } | ||
63 | else if (format == "xml") | ||
64 | { | ||
65 | Console.WriteLine("<Nini>"); | ||
66 | |||
67 | foreach (IConfig c in s.Configs) | ||
68 | { | ||
69 | int count = 0; | ||
70 | |||
71 | foreach (string k in c.GetKeys()) | ||
72 | { | ||
73 | if (k.StartsWith("Include-")) | ||
74 | continue; | ||
75 | |||
76 | count++; | ||
77 | } | ||
78 | |||
79 | if (count > 0) | ||
80 | { | ||
81 | Console.WriteLine("<Section Name=\"{0}\">", c.Name); | ||
82 | |||
83 | foreach (string k in c.GetKeys()) | ||
84 | { | ||
85 | string v = c.GetString(k); | ||
86 | |||
87 | if (k.StartsWith("Include-")) | ||
88 | continue; | ||
89 | Console.WriteLine(" <Key Name=\"{0}\" Value=\"{1}\" />", k, v); | ||
90 | } | ||
91 | |||
92 | Console.WriteLine("</Section>"); | ||
93 | } | ||
94 | } | ||
95 | Console.WriteLine("</Nini>"); | ||
96 | } | ||
97 | else if (format == "ini") | ||
98 | { | ||
99 | foreach (IConfig c in s.Configs) | ||
100 | { | ||
101 | int count = 0; | ||
102 | |||
103 | foreach (string k in c.GetKeys()) | ||
104 | { | ||
105 | if (k.StartsWith("Include-")) | ||
106 | continue; | ||
107 | |||
108 | count++; | ||
109 | } | ||
110 | |||
111 | if (count > 0) | ||
112 | { | ||
113 | Console.WriteLine("[{0}]", c.Name); | ||
114 | |||
115 | foreach (string k in c.GetKeys()) | ||
116 | { | ||
117 | string v = c.GetString(k); | ||
118 | |||
119 | if (k.StartsWith("Include-")) | ||
120 | continue; | ||
121 | Console.WriteLine("{0} = \"{1}\"", k, v); | ||
122 | } | ||
123 | |||
124 | Console.WriteLine(); | ||
125 | } | ||
126 | } | ||
127 | } | ||
128 | else | ||
129 | { | ||
130 | Console.WriteLine("Error: unknown format: {0}", format); | ||
131 | } | ||
132 | |||
133 | return 0; | ||
134 | } | ||
135 | } | ||
136 | } | ||
diff --git a/OpenSim/Tools/Configger/Properties/AssemblyInfo.cs b/OpenSim/Tools/Configger/Properties/AssemblyInfo.cs new file mode 100644 index 0000000..e88f0f5 --- /dev/null +++ b/OpenSim/Tools/Configger/Properties/AssemblyInfo.cs | |||
@@ -0,0 +1,33 @@ | |||
1 | using System.Reflection; | ||
2 | using System.Runtime.CompilerServices; | ||
3 | using System.Runtime.InteropServices; | ||
4 | |||
5 | // General Information about an assembly is controlled through the following | ||
6 | // set of attributes. Change these attribute values to modify the information | ||
7 | // associated with an assembly. | ||
8 | [assembly: AssemblyTitle("OpenSim.Tools.Configger")] | ||
9 | [assembly: AssemblyDescription("")] | ||
10 | [assembly: AssemblyConfiguration("")] | ||
11 | [assembly: AssemblyCompany("http://opensimulator.org")] | ||
12 | [assembly: AssemblyProduct("OpenSim")] | ||
13 | [assembly: AssemblyCopyright("OpenSimulator developers")] | ||
14 | [assembly: AssemblyTrademark("")] | ||
15 | [assembly: AssemblyCulture("")] | ||
16 | |||
17 | // Setting ComVisible to false makes the types in this assembly not visible | ||
18 | // to COM components. If you need to access a type in this assembly from | ||
19 | // COM, set the ComVisible attribute to true on that type. | ||
20 | [assembly: ComVisible(false)] | ||
21 | |||
22 | // The following GUID is for the ID of the typelib if this project is exposed to COM | ||
23 | [assembly: Guid("67d7fdf2-554c-40f0-8f9d-f71373c20926")] | ||
24 | |||
25 | // Version information for an assembly consists of the following four values: | ||
26 | // | ||
27 | // Major Version | ||
28 | // Minor Version | ||
29 | // Build Number | ||
30 | // Revision | ||
31 | // | ||
32 | [assembly: AssemblyVersion("0.8.2.*")] | ||
33 | |||
diff --git a/OpenSim/Tools/Configger/Util.cs b/OpenSim/Tools/Configger/Util.cs new file mode 100644 index 0000000..fe7744d --- /dev/null +++ b/OpenSim/Tools/Configger/Util.cs | |||
@@ -0,0 +1,106 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.Collections; | ||
30 | using System.Collections.Generic; | ||
31 | using System.Globalization; | ||
32 | using System.IO; | ||
33 | using System.IO.Compression; | ||
34 | using System.Net; | ||
35 | using System.Net.Sockets; | ||
36 | using System.Reflection; | ||
37 | using System.Text; | ||
38 | using System.Text.RegularExpressions; | ||
39 | using System.Threading; | ||
40 | using log4net; | ||
41 | using Nini.Config; | ||
42 | |||
43 | namespace OpenSim.Tools.Configger | ||
44 | { | ||
45 | public static class Util | ||
46 | { | ||
47 | public static string[] Glob(string path) | ||
48 | { | ||
49 | string vol=String.Empty; | ||
50 | |||
51 | if (Path.VolumeSeparatorChar != Path.DirectorySeparatorChar) | ||
52 | { | ||
53 | string[] vcomps = path.Split(new char[] {Path.VolumeSeparatorChar}, 2, StringSplitOptions.RemoveEmptyEntries); | ||
54 | |||
55 | if (vcomps.Length > 1) | ||
56 | { | ||
57 | path = vcomps[1]; | ||
58 | vol = vcomps[0]; | ||
59 | } | ||
60 | } | ||
61 | |||
62 | string[] comps = path.Split(new char[] {Path.DirectorySeparatorChar, Path.AltDirectorySeparatorChar}, StringSplitOptions.RemoveEmptyEntries); | ||
63 | |||
64 | // Glob | ||
65 | |||
66 | path = vol; | ||
67 | if (vol != String.Empty) | ||
68 | path += new String(new char[] {Path.VolumeSeparatorChar, Path.DirectorySeparatorChar}); | ||
69 | else | ||
70 | path = new String(new char[] {Path.DirectorySeparatorChar}); | ||
71 | |||
72 | List<string> paths = new List<string>(); | ||
73 | List<string> found = new List<string>(); | ||
74 | paths.Add(path); | ||
75 | |||
76 | int compIndex = -1; | ||
77 | foreach (string c in comps) | ||
78 | { | ||
79 | compIndex++; | ||
80 | |||
81 | List<string> addpaths = new List<string>(); | ||
82 | foreach (string p in paths) | ||
83 | { | ||
84 | string[] dirs = Directory.GetDirectories(p, c); | ||
85 | |||
86 | if (dirs.Length != 0) | ||
87 | { | ||
88 | foreach (string dir in dirs) | ||
89 | addpaths.Add(Path.Combine(path, dir)); | ||
90 | } | ||
91 | |||
92 | // Only add files if that is the last path component | ||
93 | if (compIndex == comps.Length - 1) | ||
94 | { | ||
95 | string[] files = Directory.GetFiles(p, c); | ||
96 | foreach (string f in files) | ||
97 | found.Add(f); | ||
98 | } | ||
99 | } | ||
100 | paths = addpaths; | ||
101 | } | ||
102 | |||
103 | return found.ToArray(); | ||
104 | } | ||
105 | } | ||
106 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient.app.skel/Contents/Info.plist b/OpenSim/Tools/LaunchSLClient/LaunchSLClient.app.skel/Contents/Info.plist new file mode 100644 index 0000000..7139350 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient.app.skel/Contents/Info.plist | |||
@@ -0,0 +1,12 @@ | |||
1 | <?xml version="1.0" encoding="UTF-8"?> | ||
2 | <!DOCTYPE plist PUBLIC "-//Apple Computer//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||
3 | <plist version="1.0"> | ||
4 | <dict> | ||
5 | <key>CFBundleIdentifier</key> | ||
6 | <string>LaunchSLClient</string> | ||
7 | <key>CFBundleExecutable</key> | ||
8 | <string>LaunchSLClient</string> | ||
9 | <key>CFBundleIconFile</key> | ||
10 | <string></string> | ||
11 | </dict> | ||
12 | </plist> | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient.app.skel/Contents/MacOS/LaunchSLClient b/OpenSim/Tools/LaunchSLClient/LaunchSLClient.app.skel/Contents/MacOS/LaunchSLClient new file mode 100755 index 0000000..22acade --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient.app.skel/Contents/MacOS/LaunchSLClient | |||
@@ -0,0 +1,13 @@ | |||
1 | #!/bin/sh | ||
2 | |||
3 | PWD=`pwd` | ||
4 | |||
5 | # Fetch the path relative to the launch point where this shell script exists. | ||
6 | APP_PATH=`echo $0 | awk '{split($0,patharr,"/"); idx=1; while(patharr[idx+3] != "") { if (patharr[idx] != "/") {printf("%s/", patharr[idx]); idx++ }} }'` | ||
7 | |||
8 | # Fetch the app name (its our own name) | ||
9 | APP_NAME=`echo $0 | awk '{split($0,patharr,"/"); idx=1; while(patharr[idx+1] != "") {idx++} printf("%s", patharr[idx]); }'` | ||
10 | |||
11 | cd "$APP_PATH/Contents/Resources" | ||
12 | |||
13 | ./$APP_NAME | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Form1.Designer.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Form1.Designer.cs new file mode 100644 index 0000000..8d07724 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Form1.Designer.cs | |||
@@ -0,0 +1,95 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | namespace LaunchSLClient | ||
29 | { | ||
30 | partial class Form1 | ||
31 | { | ||
32 | /// <summary> | ||
33 | /// Required designer variable. | ||
34 | /// </summary> | ||
35 | private System.ComponentModel.IContainer components = null; | ||
36 | |||
37 | /// <summary> | ||
38 | /// Clean up any resources being used. | ||
39 | /// </summary> | ||
40 | /// <param name="disposing">true if managed resources should be disposed; otherwise, false.</param> | ||
41 | protected override void Dispose(bool disposing) | ||
42 | { | ||
43 | if (disposing && (components != null)) | ||
44 | { | ||
45 | components.Dispose(); | ||
46 | } | ||
47 | base.Dispose(disposing); | ||
48 | } | ||
49 | |||
50 | #region Windows Form Designer generated code | ||
51 | |||
52 | /// <summary> | ||
53 | /// Required method for Designer support - do not modify | ||
54 | /// the contents of this method with the code editor. | ||
55 | /// </summary> | ||
56 | private void InitializeComponent() | ||
57 | { | ||
58 | this.comboBox1 = new System.Windows.Forms.ComboBox(); | ||
59 | this.textBox1 = new System.Windows.Forms.TextBox(); | ||
60 | this.SuspendLayout(); | ||
61 | |||
62 | this.comboBox1.FormattingEnabled = true; | ||
63 | this.comboBox1.Location = new System.Drawing.Point(37, 83); | ||
64 | this.comboBox1.Name = "comboBox1"; | ||
65 | this.comboBox1.Size = new System.Drawing.Size(348, 21); | ||
66 | this.comboBox1.TabIndex = 0; | ||
67 | this.comboBox1.Text = "Choose from list"; | ||
68 | this.comboBox1.SelectedIndexChanged += new System.EventHandler(this.comboBox1_SelectedIndexChanged); | ||
69 | |||
70 | this.textBox1.BorderStyle = System.Windows.Forms.BorderStyle.None; | ||
71 | this.textBox1.Font = new System.Drawing.Font("Microsoft Sans Serif", 12F, System.Drawing.FontStyle.Bold, System.Drawing.GraphicsUnit.Point, ((byte)(0))); | ||
72 | this.textBox1.Location = new System.Drawing.Point(37, 32); | ||
73 | this.textBox1.Name = "textBox1"; | ||
74 | this.textBox1.ReadOnly = true; | ||
75 | this.textBox1.Size = new System.Drawing.Size(292, 19); | ||
76 | this.textBox1.TabIndex = 1; | ||
77 | this.textBox1.Text = "Grid to connect to:"; | ||
78 | |||
79 | this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F); | ||
80 | this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font; | ||
81 | this.ClientSize = new System.Drawing.Size(501, 339); | ||
82 | this.Controls.Add(this.textBox1); | ||
83 | this.Controls.Add(this.comboBox1); | ||
84 | this.Name = "Form1"; | ||
85 | this.Text = "OpenSim Client Launcher"; | ||
86 | this.ResumeLayout(false); | ||
87 | this.PerformLayout(); | ||
88 | } | ||
89 | |||
90 | #endregion | ||
91 | |||
92 | private System.Windows.Forms.ComboBox comboBox1; | ||
93 | private System.Windows.Forms.TextBox textBox1; | ||
94 | } | ||
95 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Form1.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Form1.cs new file mode 100644 index 0000000..2a5d2a6 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Form1.cs | |||
@@ -0,0 +1,246 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.IO; | ||
30 | using System.Collections; | ||
31 | using System.Collections.Generic; | ||
32 | using System.ComponentModel; | ||
33 | using System.Data; | ||
34 | using System.Diagnostics; | ||
35 | using System.Drawing; | ||
36 | using System.Text; | ||
37 | using System.Text.RegularExpressions; | ||
38 | using System.Windows.Forms; | ||
39 | using Nini.Config; | ||
40 | |||
41 | namespace LaunchSLClient | ||
42 | { | ||
43 | public partial class Form1 : Form | ||
44 | { | ||
45 | string gridUrl = ""; | ||
46 | string sandboxUrl = ""; | ||
47 | string runUrl = ""; | ||
48 | string runLine = ""; | ||
49 | string exeFlags = ""; | ||
50 | string exePath = ""; | ||
51 | |||
52 | private MachineConfig m_machineConfig; | ||
53 | private List<Grid> m_grids = new List<Grid>(); | ||
54 | |||
55 | public Form1() | ||
56 | { | ||
57 | InitializeComponent(); | ||
58 | |||
59 | m_machineConfig = getMachineConfig(); | ||
60 | m_machineConfig.GetClient(ref exePath, ref runLine, ref exeFlags); | ||
61 | |||
62 | initializeGrids(); | ||
63 | |||
64 | ArrayList menuItems = new ArrayList(); | ||
65 | menuItems.Add(string.Empty); | ||
66 | |||
67 | addLocalSims(ref menuItems); | ||
68 | |||
69 | foreach (Grid grid in m_grids) | ||
70 | { | ||
71 | menuItems.Add(grid.Name + " - " + grid.URL); | ||
72 | } | ||
73 | |||
74 | comboBox1.DataSource = menuItems; | ||
75 | } | ||
76 | |||
77 | private MachineConfig getMachineConfig() | ||
78 | { | ||
79 | if (Environment.OSVersion.Platform == PlatformID.Unix) | ||
80 | { | ||
81 | if (File.Exists("/System/Library/Frameworks/Cocoa.framework/Cocoa")) | ||
82 | { | ||
83 | return new OSXConfig(); | ||
84 | } | ||
85 | else | ||
86 | { | ||
87 | return new UnixConfig(); | ||
88 | } | ||
89 | } | ||
90 | else | ||
91 | { | ||
92 | return new WindowsConfig(); | ||
93 | } | ||
94 | } | ||
95 | |||
96 | private void initializeGrids() | ||
97 | { | ||
98 | string iniFile = "LaunchSLClient.ini"; | ||
99 | |||
100 | if (!File.Exists(iniFile)) | ||
101 | return; | ||
102 | |||
103 | IniConfigSource configSource = new IniConfigSource(iniFile); | ||
104 | |||
105 | foreach (IConfig config in configSource.Configs) | ||
106 | { | ||
107 | Grid grid = new Grid(); | ||
108 | |||
109 | grid.Name = config.Name; | ||
110 | grid.LoginURI = config.GetString("loginURI", ""); | ||
111 | grid.URL = config.GetString("URL", ""); | ||
112 | |||
113 | m_grids.Add(grid); | ||
114 | } | ||
115 | } | ||
116 | |||
117 | private void addLocalSandbox(ref ArrayList menuItems) | ||
118 | { | ||
119 | // build sandbox URL from Regions/default.xml | ||
120 | // this is highly dependant on a standard default.xml | ||
121 | if (File.Exists("Regions/default.xml")) | ||
122 | { | ||
123 | string sandboxHostName = ""; | ||
124 | string sandboxPort = ""; | ||
125 | string text; | ||
126 | |||
127 | Regex myRegex = new Regex(".*internal_ip_port=\\\"(?<port>.*?)\\\".*external_host_name=\\\"(?<name>.*?)\\\".*"); | ||
128 | |||
129 | FileInfo defaultFile = new FileInfo("Regions/default.xml"); | ||
130 | StreamReader stream = defaultFile.OpenText(); | ||
131 | do | ||
132 | { | ||
133 | text = stream.ReadLine(); | ||
134 | if (text == null) | ||
135 | { | ||
136 | break; | ||
137 | } | ||
138 | MatchCollection theMatches = myRegex.Matches(text); | ||
139 | foreach (Match theMatch in theMatches) | ||
140 | { | ||
141 | if (theMatch.Length != 0) | ||
142 | { | ||
143 | sandboxHostName = theMatch.Groups["name"].ToString(); | ||
144 | sandboxPort = theMatch.Groups["port"].ToString(); | ||
145 | } | ||
146 | } | ||
147 | } while (text != null); | ||
148 | |||
149 | stream.Close(); | ||
150 | sandboxUrl = "http:\\" + sandboxHostName + ":" + sandboxPort; | ||
151 | menuItems.Add("Local Sandbox"); | ||
152 | } | ||
153 | } | ||
154 | |||
155 | private void addLocalGrid(ref ArrayList menuItems) | ||
156 | { | ||
157 | //build local grid URL from network_servers_information.xml | ||
158 | if (File.Exists("network_servers_information.xml")) | ||
159 | { | ||
160 | string text; | ||
161 | FileInfo defaultFile = new FileInfo("network_servers_information.xml"); | ||
162 | Regex myRegex = new Regex(".*UserServerURL=\\\"(?<url>.*?)\\\".*"); | ||
163 | StreamReader stream = defaultFile.OpenText(); | ||
164 | |||
165 | do | ||
166 | { | ||
167 | text = stream.ReadLine(); | ||
168 | if (text == null) | ||
169 | { | ||
170 | break; | ||
171 | } | ||
172 | foreach (Match theMatch in myRegex.Matches(text)) | ||
173 | { | ||
174 | if (theMatch.Length != 0) | ||
175 | { | ||
176 | gridUrl = theMatch.Groups["url"].ToString(); | ||
177 | } | ||
178 | } | ||
179 | } while (text != null); | ||
180 | stream.Close(); | ||
181 | if (gridUrl != null) | ||
182 | { | ||
183 | menuItems.Add("Local Grid Server"); | ||
184 | } | ||
185 | } | ||
186 | } | ||
187 | |||
188 | private void addLocalSims(ref ArrayList menuItems) | ||
189 | { | ||
190 | string configDir = m_machineConfig.GetConfigDir(); | ||
191 | |||
192 | if (!string.IsNullOrEmpty(configDir)) | ||
193 | { | ||
194 | Directory.SetCurrentDirectory(configDir); | ||
195 | |||
196 | addLocalSandbox(ref menuItems); | ||
197 | addLocalGrid(ref menuItems); | ||
198 | } | ||
199 | } | ||
200 | |||
201 | private void comboBox1_SelectedIndexChanged(object sender, EventArgs e) | ||
202 | { | ||
203 | if (comboBox1.Text == string.Empty) | ||
204 | { | ||
205 | return; | ||
206 | } | ||
207 | else if (comboBox1.Text == "Local Sandbox") | ||
208 | { | ||
209 | runUrl=" -loginuri " + sandboxUrl; | ||
210 | } | ||
211 | else if (comboBox1.Text == "Local Grid Server") | ||
212 | { | ||
213 | runUrl = " -loginuri " + gridUrl; | ||
214 | } | ||
215 | else | ||
216 | { | ||
217 | foreach (Grid grid in m_grids) | ||
218 | { | ||
219 | if (comboBox1.Text == grid.Name + " - " + grid.URL) | ||
220 | { | ||
221 | if (grid.LoginURI != string.Empty) | ||
222 | runUrl = " -loginuri " + grid.LoginURI; | ||
223 | else | ||
224 | runUrl = ""; | ||
225 | |||
226 | break; | ||
227 | } | ||
228 | } | ||
229 | } | ||
230 | |||
231 | comboBox1.DroppedDown = false; | ||
232 | Hide(); | ||
233 | |||
234 | System.Diagnostics.Process proc = new System.Diagnostics.Process(); | ||
235 | proc.StartInfo.FileName = runLine; | ||
236 | proc.StartInfo.Arguments = exeFlags + " " + runUrl; | ||
237 | proc.StartInfo.UseShellExecute = false; | ||
238 | proc.StartInfo.RedirectStandardOutput = false; | ||
239 | proc.StartInfo.WorkingDirectory = exePath; | ||
240 | proc.Start(); | ||
241 | proc.WaitForExit(); | ||
242 | |||
243 | Application.Exit(); | ||
244 | } | ||
245 | } | ||
246 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Form1.resx b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Form1.resx new file mode 100644 index 0000000..19dc0dd --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Form1.resx | |||
@@ -0,0 +1,120 @@ | |||
1 | <?xml version="1.0" encoding="utf-8"?> | ||
2 | <root> | ||
3 | <!-- | ||
4 | Microsoft ResX Schema | ||
5 | |||
6 | Version 2.0 | ||
7 | |||
8 | The primary goals of this format is to allow a simple XML format | ||
9 | that is mostly human readable. The generation and parsing of the | ||
10 | various data types are done through the TypeConverter classes | ||
11 | associated with the data types. | ||
12 | |||
13 | Example: | ||
14 | |||
15 | ... ado.net/XML headers & schema ... | ||
16 | <resheader name="resmimetype">text/microsoft-resx</resheader> | ||
17 | <resheader name="version">2.0</resheader> | ||
18 | <resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader> | ||
19 | <resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader> | ||
20 | <data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data> | ||
21 | <data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data> | ||
22 | <data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64"> | ||
23 | <value>[base64 mime encoded serialized .NET Framework object]</value> | ||
24 | </data> | ||
25 | <data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64"> | ||
26 | <value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value> | ||
27 | <comment>This is a comment</comment> | ||
28 | </data> | ||
29 | |||
30 | There are any number of "resheader" rows that contain simple | ||
31 | name/value pairs. | ||
32 | |||
33 | Each data row contains a name, and value. The row also contains a | ||
34 | type or mimetype. Type corresponds to a .NET class that support | ||
35 | text/value conversion through the TypeConverter architecture. | ||
36 | Classes that don't support this are serialized and stored with the | ||
37 | mimetype set. | ||
38 | |||
39 | The mimetype is used for serialized objects, and tells the | ||
40 | ResXResourceReader how to depersist the object. This is currently not | ||
41 | extensible. For a given mimetype the value must be set accordingly: | ||
42 | |||
43 | Note - application/x-microsoft.net.object.binary.base64 is the format | ||
44 | that the ResXResourceWriter will generate, however the reader can | ||
45 | read any of the formats listed below. | ||
46 | |||
47 | mimetype: application/x-microsoft.net.object.binary.base64 | ||
48 | value : The object must be serialized with | ||
49 | : System.Runtime.Serialization.Formatters.Binary.BinaryFormatter | ||
50 | : and then encoded with base64 encoding. | ||
51 | |||
52 | mimetype: application/x-microsoft.net.object.soap.base64 | ||
53 | value : The object must be serialized with | ||
54 | : System.Runtime.Serialization.Formatters.Soap.SoapFormatter | ||
55 | : and then encoded with base64 encoding. | ||
56 | |||
57 | mimetype: application/x-microsoft.net.object.bytearray.base64 | ||
58 | value : The object must be serialized into a byte array | ||
59 | : using a System.ComponentModel.TypeConverter | ||
60 | : and then encoded with base64 encoding. | ||
61 | --> | ||
62 | <xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata"> | ||
63 | <xsd:import namespace="http://www.w3.org/XML/1998/namespace" /> | ||
64 | <xsd:element name="root" msdata:IsDataSet="true"> | ||
65 | <xsd:complexType> | ||
66 | <xsd:choice maxOccurs="unbounded"> | ||
67 | <xsd:element name="metadata"> | ||
68 | <xsd:complexType> | ||
69 | <xsd:sequence> | ||
70 | <xsd:element name="value" type="xsd:string" minOccurs="0" /> | ||
71 | </xsd:sequence> | ||
72 | <xsd:attribute name="name" use="required" type="xsd:string" /> | ||
73 | <xsd:attribute name="type" type="xsd:string" /> | ||
74 | <xsd:attribute name="mimetype" type="xsd:string" /> | ||
75 | <xsd:attribute ref="xml:space" /> | ||
76 | </xsd:complexType> | ||
77 | </xsd:element> | ||
78 | <xsd:element name="assembly"> | ||
79 | <xsd:complexType> | ||
80 | <xsd:attribute name="alias" type="xsd:string" /> | ||
81 | <xsd:attribute name="name" type="xsd:string" /> | ||
82 | </xsd:complexType> | ||
83 | </xsd:element> | ||
84 | <xsd:element name="data"> | ||
85 | <xsd:complexType> | ||
86 | <xsd:sequence> | ||
87 | <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | ||
88 | <xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" /> | ||
89 | </xsd:sequence> | ||
90 | <xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" /> | ||
91 | <xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" /> | ||
92 | <xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" /> | ||
93 | <xsd:attribute ref="xml:space" /> | ||
94 | </xsd:complexType> | ||
95 | </xsd:element> | ||
96 | <xsd:element name="resheader"> | ||
97 | <xsd:complexType> | ||
98 | <xsd:sequence> | ||
99 | <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | ||
100 | </xsd:sequence> | ||
101 | <xsd:attribute name="name" type="xsd:string" use="required" /> | ||
102 | </xsd:complexType> | ||
103 | </xsd:element> | ||
104 | </xsd:choice> | ||
105 | </xsd:complexType> | ||
106 | </xsd:element> | ||
107 | </xsd:schema> | ||
108 | <resheader name="resmimetype"> | ||
109 | <value>text/microsoft-resx</value> | ||
110 | </resheader> | ||
111 | <resheader name="version"> | ||
112 | <value>2.0</value> | ||
113 | </resheader> | ||
114 | <resheader name="reader"> | ||
115 | <value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | ||
116 | </resheader> | ||
117 | <resheader name="writer"> | ||
118 | <value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | ||
119 | </resheader> | ||
120 | </root> \ No newline at end of file | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Grid.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Grid.cs new file mode 100644 index 0000000..58ebbf0 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Grid.cs | |||
@@ -0,0 +1,38 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | |||
30 | namespace LaunchSLClient | ||
31 | { | ||
32 | public struct Grid | ||
33 | { | ||
34 | public string Name; | ||
35 | public string LoginURI; | ||
36 | public string URL; | ||
37 | } | ||
38 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/LauncherException.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/LauncherException.cs new file mode 100644 index 0000000..a28eb20 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/LauncherException.cs | |||
@@ -0,0 +1,52 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.Collections.Generic; | ||
30 | using System.Text; | ||
31 | |||
32 | namespace LaunchSLClient | ||
33 | { | ||
34 | class LauncherException : ApplicationException | ||
35 | { | ||
36 | private const string CUSTOMMESSAGE = "The SL Client Launcher has failed with the following error: "; | ||
37 | |||
38 | private LauncherException() { } | ||
39 | |||
40 | public LauncherException(string errorMesssage, string source) | ||
41 | : base (CUSTOMMESSAGE + errorMesssage) | ||
42 | { | ||
43 | base.Source = source; | ||
44 | } | ||
45 | |||
46 | public LauncherException(string errorMessage, string source, Exception innerException) | ||
47 | : base(CUSTOMMESSAGE + errorMessage, innerException) | ||
48 | { | ||
49 | base.Source = source; | ||
50 | } | ||
51 | } | ||
52 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/MachineConfig.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/MachineConfig.cs new file mode 100644 index 0000000..6020347 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/MachineConfig.cs | |||
@@ -0,0 +1,38 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.IO; | ||
30 | |||
31 | namespace LaunchSLClient | ||
32 | { | ||
33 | public abstract class MachineConfig | ||
34 | { | ||
35 | public abstract void GetClient(ref string exePath, ref string runLine, ref string exeFlags); | ||
36 | public abstract string GetConfigDir(); | ||
37 | } | ||
38 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/OSXConfig.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/OSXConfig.cs new file mode 100644 index 0000000..19a2e54 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/OSXConfig.cs | |||
@@ -0,0 +1,55 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.IO; | ||
30 | |||
31 | namespace LaunchSLClient | ||
32 | { | ||
33 | public class OSXConfig : UnixConfig | ||
34 | { | ||
35 | public override void GetClient(ref string exePath, ref string runLine, ref string exeFlags) | ||
36 | { | ||
37 | if (Directory.Exists("/Applications/Second Life.app")) | ||
38 | { | ||
39 | exePath = "/Applications/Second Life.app/Contents/MacOS"; | ||
40 | } | ||
41 | else if (Directory.Exists("/Applications/Second Life Release Candidate.app")) | ||
42 | { | ||
43 | exePath = "/Applications/Second Life Release Candidate.app/Contents/MacOS"; | ||
44 | } | ||
45 | |||
46 | runLine = Path.Combine(exePath, "Second Life"); | ||
47 | exeFlags = string.Empty; | ||
48 | } | ||
49 | |||
50 | public override string GetConfigDir() | ||
51 | { | ||
52 | return string.Empty; | ||
53 | } | ||
54 | } | ||
55 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Program.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Program.cs new file mode 100644 index 0000000..bfd43cf --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Program.cs | |||
@@ -0,0 +1,51 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.Collections.Generic; | ||
30 | using System.Windows.Forms; | ||
31 | |||
32 | namespace LaunchSLClient | ||
33 | { | ||
34 | static class Program | ||
35 | { | ||
36 | [STAThread] | ||
37 | static void Main() | ||
38 | { | ||
39 | try | ||
40 | { | ||
41 | Application.EnableVisualStyles(); | ||
42 | Application.SetCompatibleTextRenderingDefault(false); | ||
43 | Application.Run(new Form1()); | ||
44 | } | ||
45 | catch (Exception ex) | ||
46 | { | ||
47 | MessageBox.Show(ex.ToString(), "Error"); | ||
48 | } | ||
49 | } | ||
50 | } | ||
51 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/AssemblyInfo.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/AssemblyInfo.cs new file mode 100644 index 0000000..f6a0c64 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/AssemblyInfo.cs | |||
@@ -0,0 +1,60 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System.Reflection; | ||
29 | using System.Runtime.CompilerServices; | ||
30 | using System.Runtime.InteropServices; | ||
31 | |||
32 | // General information about an assembly is controlled through the following | ||
33 | // set of attributes. Change these attribute values to modify the information | ||
34 | // associated with an assembly. | ||
35 | [assembly: AssemblyTitle("LaunchSLClient")] | ||
36 | [assembly: AssemblyDescription("")] | ||
37 | [assembly: AssemblyConfiguration("")] | ||
38 | [assembly: AssemblyCompany("http://opensimulator.org")] | ||
39 | [assembly: AssemblyProduct("LaunchSLClient")] | ||
40 | [assembly: AssemblyCopyright("Copyright (c) 2007")] | ||
41 | [assembly: AssemblyTrademark("")] | ||
42 | [assembly: AssemblyCulture("")] | ||
43 | |||
44 | // Setting ComVisible to false makes the types in this assembly not visible | ||
45 | // to COM components. If you need to access a type in this assembly from | ||
46 | // COM, set the ComVisible attribute to true on that type. | ||
47 | [assembly: ComVisible(false)] | ||
48 | |||
49 | // The following GUID is for the ID of the typelib if this project is exposed to COM | ||
50 | [assembly: Guid("b08c6904-e6cc-4d9c-8d24-feb0464b1648")] | ||
51 | |||
52 | // Version information for an assembly consists of the following four values: | ||
53 | // | ||
54 | // Major Version | ||
55 | // Minor Version | ||
56 | // Build Number | ||
57 | // Revision | ||
58 | // | ||
59 | [assembly: AssemblyVersion("0.6.3.*")] | ||
60 | [assembly: AssemblyFileVersion("1.0.0.0")] | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Resources.Designer.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Resources.Designer.cs new file mode 100644 index 0000000..6b9a8a1 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Resources.Designer.cs | |||
@@ -0,0 +1,95 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | //------------------------------------------------------------------------------ | ||
29 | // <auto-generated> | ||
30 | // This code was generated by a tool. | ||
31 | // Runtime Version:2.0.50727.832 | ||
32 | // | ||
33 | // Changes to this file may cause incorrect behavior and will be lost if | ||
34 | // the code is regenerated. | ||
35 | // </auto-generated> | ||
36 | //------------------------------------------------------------------------------ | ||
37 | |||
38 | namespace LaunchSLClient.Properties | ||
39 | { | ||
40 | /// <summary> | ||
41 | /// A strongly-typed resource class, for looking up localized strings, etc. | ||
42 | /// </summary> | ||
43 | // This class was auto-generated by the StronglyTypedResourceBuilder | ||
44 | // class via a tool like ResGen or Visual Studio. | ||
45 | // To add or remove a member, edit your .ResX file then rerun ResGen | ||
46 | // with the /str option, or rebuild your VS project. | ||
47 | [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "2.0.0.0")] | ||
48 | [global::System.Diagnostics.DebuggerNonUserCodeAttribute()] | ||
49 | [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] | ||
50 | internal class Resources | ||
51 | { | ||
52 | private static global::System.Resources.ResourceManager resourceMan; | ||
53 | |||
54 | private static global::System.Globalization.CultureInfo resourceCulture; | ||
55 | |||
56 | [global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")] | ||
57 | internal Resources() | ||
58 | { | ||
59 | } | ||
60 | |||
61 | /// <summary> | ||
62 | /// Returns the cached ResourceManager instance used by this class. | ||
63 | /// </summary> | ||
64 | [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] | ||
65 | internal static global::System.Resources.ResourceManager ResourceManager | ||
66 | { | ||
67 | get | ||
68 | { | ||
69 | if ((resourceMan == null)) | ||
70 | { | ||
71 | global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("LaunchSLClient.Properties.Resources", typeof(Resources).Assembly); | ||
72 | resourceMan = temp; | ||
73 | } | ||
74 | return resourceMan; | ||
75 | } | ||
76 | } | ||
77 | |||
78 | /// <summary> | ||
79 | /// Overrides the current thread's CurrentUICulture property for all | ||
80 | /// resource lookups using this strongly typed resource class. | ||
81 | /// </summary> | ||
82 | [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] | ||
83 | internal static global::System.Globalization.CultureInfo Culture | ||
84 | { | ||
85 | get | ||
86 | { | ||
87 | return resourceCulture; | ||
88 | } | ||
89 | set | ||
90 | { | ||
91 | resourceCulture = value; | ||
92 | } | ||
93 | } | ||
94 | } | ||
95 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Resources.resx b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Resources.resx new file mode 100644 index 0000000..af7dbeb --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Resources.resx | |||
@@ -0,0 +1,117 @@ | |||
1 | <?xml version="1.0" encoding="utf-8"?> | ||
2 | <root> | ||
3 | <!-- | ||
4 | Microsoft ResX Schema | ||
5 | |||
6 | Version 2.0 | ||
7 | |||
8 | The primary goals of this format is to allow a simple XML format | ||
9 | that is mostly human readable. The generation and parsing of the | ||
10 | various data types are done through the TypeConverter classes | ||
11 | associated with the data types. | ||
12 | |||
13 | Example: | ||
14 | |||
15 | ... ado.net/XML headers & schema ... | ||
16 | <resheader name="resmimetype">text/microsoft-resx</resheader> | ||
17 | <resheader name="version">2.0</resheader> | ||
18 | <resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader> | ||
19 | <resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader> | ||
20 | <data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data> | ||
21 | <data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data> | ||
22 | <data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64"> | ||
23 | <value>[base64 mime encoded serialized .NET Framework object]</value> | ||
24 | </data> | ||
25 | <data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64"> | ||
26 | <value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value> | ||
27 | <comment>This is a comment</comment> | ||
28 | </data> | ||
29 | |||
30 | There are any number of "resheader" rows that contain simple | ||
31 | name/value pairs. | ||
32 | |||
33 | Each data row contains a name, and value. The row also contains a | ||
34 | type or mimetype. Type corresponds to a .NET class that support | ||
35 | text/value conversion through the TypeConverter architecture. | ||
36 | Classes that don't support this are serialized and stored with the | ||
37 | mimetype set. | ||
38 | |||
39 | The mimetype is used for serialized objects, and tells the | ||
40 | ResXResourceReader how to depersist the object. This is currently not | ||
41 | extensible. For a given mimetype the value must be set accordingly: | ||
42 | |||
43 | Note - application/x-microsoft.net.object.binary.base64 is the format | ||
44 | that the ResXResourceWriter will generate, however the reader can | ||
45 | read any of the formats listed below. | ||
46 | |||
47 | mimetype: application/x-microsoft.net.object.binary.base64 | ||
48 | value : The object must be serialized with | ||
49 | : System.Serialization.Formatters.Binary.BinaryFormatter | ||
50 | : and then encoded with base64 encoding. | ||
51 | |||
52 | mimetype: application/x-microsoft.net.object.soap.base64 | ||
53 | value : The object must be serialized with | ||
54 | : System.Runtime.Serialization.Formatters.Soap.SoapFormatter | ||
55 | : and then encoded with base64 encoding. | ||
56 | |||
57 | mimetype: application/x-microsoft.net.object.bytearray.base64 | ||
58 | value : The object must be serialized into a byte array | ||
59 | : using a System.ComponentModel.TypeConverter | ||
60 | : and then encoded with base64 encoding. | ||
61 | --> | ||
62 | <xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata"> | ||
63 | <xsd:element name="root" msdata:IsDataSet="true"> | ||
64 | <xsd:complexType> | ||
65 | <xsd:choice maxOccurs="unbounded"> | ||
66 | <xsd:element name="metadata"> | ||
67 | <xsd:complexType> | ||
68 | <xsd:sequence> | ||
69 | <xsd:element name="value" type="xsd:string" minOccurs="0" /> | ||
70 | </xsd:sequence> | ||
71 | <xsd:attribute name="name" type="xsd:string" /> | ||
72 | <xsd:attribute name="type" type="xsd:string" /> | ||
73 | <xsd:attribute name="mimetype" type="xsd:string" /> | ||
74 | </xsd:complexType> | ||
75 | </xsd:element> | ||
76 | <xsd:element name="assembly"> | ||
77 | <xsd:complexType> | ||
78 | <xsd:attribute name="alias" type="xsd:string" /> | ||
79 | <xsd:attribute name="name" type="xsd:string" /> | ||
80 | </xsd:complexType> | ||
81 | </xsd:element> | ||
82 | <xsd:element name="data"> | ||
83 | <xsd:complexType> | ||
84 | <xsd:sequence> | ||
85 | <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | ||
86 | <xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" /> | ||
87 | </xsd:sequence> | ||
88 | <xsd:attribute name="name" type="xsd:string" msdata:Ordinal="1" /> | ||
89 | <xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" /> | ||
90 | <xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" /> | ||
91 | </xsd:complexType> | ||
92 | </xsd:element> | ||
93 | <xsd:element name="resheader"> | ||
94 | <xsd:complexType> | ||
95 | <xsd:sequence> | ||
96 | <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | ||
97 | </xsd:sequence> | ||
98 | <xsd:attribute name="name" type="xsd:string" use="required" /> | ||
99 | </xsd:complexType> | ||
100 | </xsd:element> | ||
101 | </xsd:choice> | ||
102 | </xsd:complexType> | ||
103 | </xsd:element> | ||
104 | </xsd:schema> | ||
105 | <resheader name="resmimetype"> | ||
106 | <value>text/microsoft-resx</value> | ||
107 | </resheader> | ||
108 | <resheader name="version"> | ||
109 | <value>2.0</value> | ||
110 | </resheader> | ||
111 | <resheader name="reader"> | ||
112 | <value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | ||
113 | </resheader> | ||
114 | <resheader name="writer"> | ||
115 | <value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | ||
116 | </resheader> | ||
117 | </root> \ No newline at end of file | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Settings.Designer.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Settings.Designer.cs new file mode 100644 index 0000000..5ae7be2 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Settings.Designer.cs | |||
@@ -0,0 +1,54 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | //------------------------------------------------------------------------------ | ||
29 | // <auto-generated> | ||
30 | // This code was generated by a tool. | ||
31 | // Runtime Version:2.0.50727.832 | ||
32 | // | ||
33 | // Changes to this file may cause incorrect behavior and will be lost if | ||
34 | // the code is regenerated. | ||
35 | // </auto-generated> | ||
36 | //------------------------------------------------------------------------------ | ||
37 | |||
38 | namespace LaunchSLClient.Properties | ||
39 | { | ||
40 | [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] | ||
41 | [global::System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.VisualStudio.Editors.SettingsDesigner.SettingsSingleFileGenerator", "8.0.0.0")] | ||
42 | internal sealed partial class Settings : global::System.Configuration.ApplicationSettingsBase | ||
43 | { | ||
44 | private static Settings defaultInstance = ((Settings)(global::System.Configuration.ApplicationSettingsBase.Synchronized(new Settings()))); | ||
45 | |||
46 | public static Settings Default | ||
47 | { | ||
48 | get | ||
49 | { | ||
50 | return defaultInstance; | ||
51 | } | ||
52 | } | ||
53 | } | ||
54 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Settings.settings b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Settings.settings new file mode 100644 index 0000000..3964565 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/Properties/Settings.settings | |||
@@ -0,0 +1,7 @@ | |||
1 | <?xml version='1.0' encoding='utf-8'?> | ||
2 | <SettingsFile xmlns="http://schemas.microsoft.com/VisualStudio/2004/01/settings" CurrentProfile="(Default)"> | ||
3 | <Profiles> | ||
4 | <Profile Name="(Default)" /> | ||
5 | </Profiles> | ||
6 | <Settings /> | ||
7 | </SettingsFile> | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/UnixConfig.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/UnixConfig.cs new file mode 100644 index 0000000..5895dad --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/UnixConfig.cs | |||
@@ -0,0 +1,57 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.IO; | ||
30 | |||
31 | namespace LaunchSLClient | ||
32 | { | ||
33 | public class UnixConfig : MachineConfig | ||
34 | { | ||
35 | public override void GetClient(ref string exePath, ref string runLine, ref string exeFlags) | ||
36 | { | ||
37 | exePath = string.Empty; | ||
38 | exeFlags = string.Empty; | ||
39 | |||
40 | foreach (string path in Environment.GetEnvironmentVariable("PATH").Split(':')) | ||
41 | { | ||
42 | if (File.Exists(Path.Combine(path, "secondlife"))) | ||
43 | { | ||
44 | exePath = path; | ||
45 | break; | ||
46 | } | ||
47 | } | ||
48 | |||
49 | runLine = Path.Combine(exePath, "secondlife"); | ||
50 | } | ||
51 | |||
52 | public override string GetConfigDir() | ||
53 | { | ||
54 | return string.Empty; | ||
55 | } | ||
56 | } | ||
57 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/LaunchSLClient/WindowsConfig.cs b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/WindowsConfig.cs new file mode 100644 index 0000000..ac0186d --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/LaunchSLClient/WindowsConfig.cs | |||
@@ -0,0 +1,69 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.IO; | ||
30 | using Microsoft.Win32; | ||
31 | |||
32 | namespace LaunchSLClient | ||
33 | { | ||
34 | public class WindowsConfig : MachineConfig | ||
35 | { | ||
36 | public override void GetClient(ref string exePath, ref string runLine, ref string exeFlags) | ||
37 | { | ||
38 | RegistryKey regKey = Registry.LocalMachine.OpenSubKey(@"SOFTWARE\Linden Research, Inc.\SecondLife"); | ||
39 | if (regKey == null) | ||
40 | { | ||
41 | regKey = Registry.LocalMachine.OpenSubKey(@"SOFTWARE\Wow6432Node\Linden Research, Inc.\SecondLife"); | ||
42 | if (regKey == null) | ||
43 | { | ||
44 | throw new LauncherException("Can't find Second Life. Are you sure it is installed?", "LauncherException.Form1"); | ||
45 | } | ||
46 | } | ||
47 | string exe = regKey.GetValue("Exe").ToString(); | ||
48 | exeFlags = regKey.GetValue("Flags").ToString(); | ||
49 | exePath = regKey.GetValue("").ToString(); | ||
50 | runLine = Path.Combine(exePath, exe); | ||
51 | Registry.LocalMachine.Flush(); | ||
52 | Registry.LocalMachine.Close(); | ||
53 | } | ||
54 | |||
55 | public override string GetConfigDir() | ||
56 | { | ||
57 | RegistryKey key = Registry.LocalMachine.OpenSubKey(@"SOFTWARE\OpenSim\OpenSim"); | ||
58 | |||
59 | if (key == null) | ||
60 | { | ||
61 | return string.Empty; | ||
62 | } | ||
63 | else | ||
64 | { | ||
65 | return key.GetValue("Path").ToString(); | ||
66 | } | ||
67 | } | ||
68 | } | ||
69 | } | ||
diff --git a/OpenSim/Tools/LaunchSLClient/make-OSX-app.sh b/OpenSim/Tools/LaunchSLClient/make-OSX-app.sh new file mode 100755 index 0000000..aa37cb6 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/make-OSX-app.sh | |||
@@ -0,0 +1,40 @@ | |||
1 | #!/bin/sh | ||
2 | |||
3 | # This script will build LaunchSLClient.app from the .exe, .dll's, and | ||
4 | # other necessary files. | ||
5 | # | ||
6 | # This should be run from the bin directory. | ||
7 | |||
8 | APP_NAME="LaunchSLClient" | ||
9 | SOURCE_PATH="../OpenSim/Tools/${APP_NAME}" | ||
10 | |||
11 | ASSEMBLIES="mscorlib.dll \ | ||
12 | System.Windows.Forms.dll \ | ||
13 | System.Drawing.dll \ | ||
14 | System.Configuration.dll \ | ||
15 | System.Xml.dll \ | ||
16 | System.Security.dll \ | ||
17 | Mono.Security.dll \ | ||
18 | System.Data.dll \ | ||
19 | Mono.Data.Tds.dll \ | ||
20 | System.Transactions.dll \ | ||
21 | System.EnterpriseServices.dll \ | ||
22 | Mono.Mozilla.dll \ | ||
23 | Mono.Posix.dll \ | ||
24 | Accessibility.dll" | ||
25 | |||
26 | if [ ! -e ${APP_NAME}.exe ]; then | ||
27 | echo "Error: Could not find ${APP_NAME}.exe." >& 2 | ||
28 | echo "Have you built it, and are you currently in the bin directory?" >& 2 | ||
29 | exit 1 | ||
30 | fi | ||
31 | |||
32 | mkbundle2 -z -o ${APP_NAME} ${APP_NAME}.exe ${ASSEMBLIES} || exit 1 | ||
33 | |||
34 | if [ -d ${APP_NAME}.app ]; then rm -rf ${APP_NAME}.app; fi | ||
35 | cp -r ${SOURCE_PATH}/${APP_NAME}.app.skel ${APP_NAME}.app | ||
36 | |||
37 | # mkbundle doesn't seem to recognize the -L option, so we can't include Nini.dll in the bundling | ||
38 | cp Nini.dll ${APP_NAME}.app/Contents/Resources | ||
39 | |||
40 | cp ${APP_NAME} ${APP_NAME}.ini ${APP_NAME}.app/Contents/Resources | ||
diff --git a/OpenSim/Tools/LaunchSLClient/prebuild.xml b/OpenSim/Tools/LaunchSLClient/prebuild.xml new file mode 100644 index 0000000..4a13365 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/prebuild.xml | |||
@@ -0,0 +1,67 @@ | |||
1 | <?xml version="1.0" encoding="utf-8" ?> | ||
2 | <Prebuild xmlns="http://dnpb.sourceforge.net/schemas/prebuild-1.7.xsd" version="1.7"> | ||
3 | <Solution name="LaunchSLClient" activeConfig="Debug" path="./" version="0.5.0-svn"> | ||
4 | <Configuration name="Debug"> | ||
5 | <Options> | ||
6 | <CompilerDefines>TRACE;DEBUG</CompilerDefines> | ||
7 | <OptimizeCode>false</OptimizeCode> | ||
8 | <CheckUnderflowOverflow>false</CheckUnderflowOverflow> | ||
9 | <AllowUnsafe>false</AllowUnsafe> | ||
10 | <WarningLevel>4</WarningLevel> | ||
11 | <WarningsAsErrors>false</WarningsAsErrors> | ||
12 | <SuppressWarnings></SuppressWarnings> | ||
13 | <OutputPath>../../../bin</OutputPath> | ||
14 | <DebugInformation>true</DebugInformation> | ||
15 | <IncrementalBuild>true</IncrementalBuild> | ||
16 | <NoStdLib>false</NoStdLib> | ||
17 | </Options> | ||
18 | </Configuration> | ||
19 | <Configuration name="Release"> | ||
20 | <Options> | ||
21 | <CompilerDefines>TRACE</CompilerDefines> | ||
22 | <OptimizeCode>true</OptimizeCode> | ||
23 | <CheckUnderflowOverflow>false</CheckUnderflowOverflow> | ||
24 | <AllowUnsafe>false</AllowUnsafe> | ||
25 | <WarningLevel>4</WarningLevel> | ||
26 | <WarningsAsErrors>false</WarningsAsErrors> | ||
27 | <SuppressWarnings></SuppressWarnings> | ||
28 | <OutputPath>../../../bin</OutputPath> | ||
29 | <DebugInformation>false</DebugInformation> | ||
30 | <IncrementalBuild>true</IncrementalBuild> | ||
31 | <NoStdLib>false</NoStdLib> | ||
32 | </Options> | ||
33 | </Configuration> | ||
34 | |||
35 | <Project name="LaunchSLClient" path="LaunchSLClient" type="Exe"> | ||
36 | <Configuration name="Debug"> | ||
37 | <Options> | ||
38 | <OutputPath>../../../../bin/</OutputPath> | ||
39 | </Options> | ||
40 | </Configuration> | ||
41 | <Configuration name="Release"> | ||
42 | <Options> | ||
43 | <OutputPath>../../../../bin/</OutputPath> | ||
44 | </Options> | ||
45 | </Configuration> | ||
46 | |||
47 | <ReferencePath>../../../../bin/</ReferencePath> | ||
48 | <Reference name="System" localCopy="false"/> | ||
49 | <Reference name="System.IO" localCopy="false"/> | ||
50 | <Reference name="System.Collections" localCopy="false"/> | ||
51 | <Reference name="System.Collections.Generic" localCopy="false"/> | ||
52 | <Reference name="System.ComponentModel" localCopy="false"/> | ||
53 | <Reference name="System.Data" localCopy="false"/> | ||
54 | <Reference name="System.Diagnostics" localCopy="false"/> | ||
55 | <Reference name="System.Drawing" localCopy="false"/> | ||
56 | <Reference name="System.Text" localCopy="false"/> | ||
57 | <Reference name="System.Text.RegularExpressions" localCopy="false"/> | ||
58 | <Reference name="System.Windows.Forms" localCopy="false"/> | ||
59 | <Reference name="Microsoft.Win32" localCopy="false"/> | ||
60 | <Reference name="Nini.dll"/> | ||
61 | |||
62 | <Files> | ||
63 | <Match pattern="*.cs" recurse="true"/> | ||
64 | </Files> | ||
65 | </Project> | ||
66 | </Solution> | ||
67 | </Prebuild> | ||
diff --git a/OpenSim/Tools/LaunchSLClient/runprebuild.bat b/OpenSim/Tools/LaunchSLClient/runprebuild.bat new file mode 100755 index 0000000..41e2714 --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/runprebuild.bat | |||
@@ -0,0 +1,4 @@ | |||
1 | ..\..\..\bin\Prebuild.exe /target nant | ||
2 | ..\..\..\bin\Prebuild.exe /target vs2008 | ||
3 | echo C:\WINDOWS\Microsoft.NET\Framework\v3.5\msbuild LaunchSLClient.sln > compile.bat | ||
4 | |||
diff --git a/OpenSim/Tools/LaunchSLClient/runprebuild.sh b/OpenSim/Tools/LaunchSLClient/runprebuild.sh new file mode 100755 index 0000000..25ea4cd --- /dev/null +++ b/OpenSim/Tools/LaunchSLClient/runprebuild.sh | |||
@@ -0,0 +1,5 @@ | |||
1 | #!/bin/sh | ||
2 | |||
3 | mono ../../../bin/Prebuild.exe /target nant | ||
4 | mono ../../../bin/Prebuild.exe /target monodev | ||
5 | mono ../../../bin/Prebuild.exe /target vs2008 | ||
diff --git a/OpenSim/Tools/classaudit.pl b/OpenSim/Tools/classaudit.pl new file mode 100755 index 0000000..d5d28c7 --- /dev/null +++ b/OpenSim/Tools/classaudit.pl | |||
@@ -0,0 +1,133 @@ | |||
1 | #!/usr/bin/perl | ||
2 | # | ||
3 | # Audit tool for OpenSim class and namespace definitions. | ||
4 | # | ||
5 | # Copyright 2007 IBM | ||
6 | # | ||
7 | # Authors: Sean Dague | ||
8 | # | ||
9 | # Redistribution and use in source and binary forms, with or without | ||
10 | # modification, are permitted provided that the following conditions are met: | ||
11 | # * Redistributions of source code must retain the above copyright | ||
12 | # notice, this list of conditions and the following disclaimer. | ||
13 | # * Redistributions in binary form must reproduce the above copyright | ||
14 | # notice, this list of conditions and the following disclaimer in the | ||
15 | # documentation and/or other materials provided with the distribution. | ||
16 | # * Neither the name of the OpenSim Project nor the | ||
17 | # names of its contributors may be used to endorse or promote products | ||
18 | # derived from this software without specific prior written permission. | ||
19 | # | ||
20 | # THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
21 | # EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
22 | # WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
23 | # DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
24 | # DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
25 | # (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
26 | # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
27 | # ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
28 | # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
29 | # SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
30 | |||
31 | use strict; | ||
32 | use File::Find; | ||
33 | use Data::Dumper; | ||
34 | use constant YELLOW => "\033[33m"; | ||
35 | use constant RED => "\033[31m"; | ||
36 | use constant CLEAR => "\033[0m"; | ||
37 | our %totals; | ||
38 | |||
39 | |||
40 | find(\&test, "../../OpenSim"); | ||
41 | print Dumper(\%totals); | ||
42 | |||
43 | sub test { | ||
44 | my $file = $File::Find::name; | ||
45 | my $dir = $File::Find::dir; | ||
46 | $file =~ s{^../../}{}; #strip off prefix | ||
47 | $dir =~ s{^../../}{}; #strip off prefix | ||
48 | |||
49 | return if ($file !~ /\.cs$/); | ||
50 | return if ($file =~ /AssemblyInfo\.cs$/); | ||
51 | |||
52 | print "Processing File: $file\n"; | ||
53 | |||
54 | my $namespace = find_namespace($_); | ||
55 | my $class = find_class($_); | ||
56 | |||
57 | |||
58 | |||
59 | if(cmp_namespace($namespace, $dir) == 1) { | ||
60 | $totals{goodns}++; | ||
61 | } else { | ||
62 | $totals{badns}++; | ||
63 | } | ||
64 | |||
65 | |||
66 | if(cmp_class($namespace, $class, $file) == 1) { | ||
67 | $totals{goodclass}++; | ||
68 | } else { | ||
69 | $totals{badclass}++; | ||
70 | } | ||
71 | print "\n"; | ||
72 | } | ||
73 | |||
74 | sub find_class { | ||
75 | my $file = shift; | ||
76 | my $content = slurp($file); | ||
77 | if ($content =~ /\n\s*(public|private|protected)?\s*(class|interface)\s+(\S+)/) { | ||
78 | return $3; | ||
79 | } | ||
80 | return ""; | ||
81 | } | ||
82 | |||
83 | sub find_namespace { | ||
84 | my $file = shift; | ||
85 | my $content = slurp($file); | ||
86 | |||
87 | if ($content =~ /\bnamespace\s+(\S+)/s) { | ||
88 | return $1; | ||
89 | } | ||
90 | return ""; | ||
91 | } | ||
92 | |||
93 | sub slurp { | ||
94 | my $file = shift; | ||
95 | local(*IN); | ||
96 | local $/ = undef; | ||
97 | |||
98 | open(IN, "$file") or die "Can't open '$file': $!"; | ||
99 | my $content = <IN>; | ||
100 | close(IN); | ||
101 | |||
102 | return $content; | ||
103 | } | ||
104 | |||
105 | sub cmp_class { | ||
106 | my ($ns, $class, $file) = @_; | ||
107 | $class = "$ns.$class"; | ||
108 | my $classtrans = $class; | ||
109 | $classtrans =~ s{\.}{/}g; | ||
110 | $classtrans .= ".cs"; | ||
111 | |||
112 | if($classtrans ne $file) { | ||
113 | error(YELLOW, "CLASS: $class != $file"); | ||
114 | return -1; | ||
115 | } | ||
116 | return 1; | ||
117 | } | ||
118 | |||
119 | sub cmp_namespace { | ||
120 | my ($ns, $dir) = @_; | ||
121 | my $nstrans = $ns; | ||
122 | $nstrans =~ s{\.}{/}g; | ||
123 | |||
124 | if($nstrans ne $dir) { | ||
125 | error(RED, "NS: $ns != $dir"); | ||
126 | return -1; | ||
127 | } | ||
128 | return 1; | ||
129 | } | ||
130 | |||
131 | sub error { | ||
132 | print @_, CLEAR, "\n"; | ||
133 | } | ||
diff --git a/OpenSim/Tools/pCampBot/Behaviours/AbstractBehaviour.cs b/OpenSim/Tools/pCampBot/Behaviours/AbstractBehaviour.cs new file mode 100644 index 0000000..c1ba36b --- /dev/null +++ b/OpenSim/Tools/pCampBot/Behaviours/AbstractBehaviour.cs | |||
@@ -0,0 +1,64 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using OpenMetaverse; | ||
29 | using System; | ||
30 | using System.Collections.Generic; | ||
31 | using System.Linq; | ||
32 | using System.Threading; | ||
33 | using pCampBot.Interfaces; | ||
34 | |||
35 | namespace pCampBot | ||
36 | { | ||
37 | public abstract class AbstractBehaviour : IBehaviour | ||
38 | { | ||
39 | /// <summary> | ||
40 | /// Abbreviated name of this behaviour. | ||
41 | /// </summary> | ||
42 | public string AbbreviatedName { get; protected set; } | ||
43 | |||
44 | public string Name { get; protected set; } | ||
45 | |||
46 | public Bot Bot { get; protected set; } | ||
47 | |||
48 | public abstract void Action(); | ||
49 | |||
50 | public virtual void Interrupt() {} | ||
51 | |||
52 | protected AutoResetEvent m_interruptEvent = new AutoResetEvent(false); | ||
53 | |||
54 | public virtual void Initialize(Bot bot) | ||
55 | { | ||
56 | Bot = bot; | ||
57 | } | ||
58 | |||
59 | public virtual void Close() | ||
60 | { | ||
61 | Interrupt(); | ||
62 | } | ||
63 | } | ||
64 | } | ||
diff --git a/OpenSim/Tools/pCampBot/Behaviours/CrossBehaviour.cs b/OpenSim/Tools/pCampBot/Behaviours/CrossBehaviour.cs new file mode 100644 index 0000000..4d806fc --- /dev/null +++ b/OpenSim/Tools/pCampBot/Behaviours/CrossBehaviour.cs | |||
@@ -0,0 +1,173 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.Collections.Generic; | ||
30 | using System.Reflection; | ||
31 | using System.Threading; | ||
32 | using log4net; | ||
33 | using OpenMetaverse; | ||
34 | using OpenSim.Framework; | ||
35 | using pCampBot.Interfaces; | ||
36 | |||
37 | namespace pCampBot | ||
38 | { | ||
39 | /// <summary> | ||
40 | /// Get the bot to make a region crossing. | ||
41 | /// </summary> | ||
42 | public class CrossBehaviour : AbstractBehaviour | ||
43 | { | ||
44 | private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); | ||
45 | |||
46 | public AutoResetEvent m_regionCrossedMutex = new AutoResetEvent(false); | ||
47 | |||
48 | public const int m_regionCrossingTimeout = 1000 * 60; | ||
49 | |||
50 | public CrossBehaviour() | ||
51 | { | ||
52 | AbbreviatedName = "c"; | ||
53 | Name = "Cross"; | ||
54 | } | ||
55 | |||
56 | public override void Action() | ||
57 | { | ||
58 | GridClient client = Bot.Client; | ||
59 | |||
60 | // // Fly to make the border cross easier. | ||
61 | // client.Self.Movement.Fly = true; | ||
62 | // client.Self.Movement.Fly = false; | ||
63 | |||
64 | // Seek out neighbouring region | ||
65 | Simulator currentSim = client.Network.CurrentSim; | ||
66 | ulong currentHandle = currentSim.Handle; | ||
67 | uint currentX, currentY; | ||
68 | Utils.LongToUInts(currentHandle, out currentX, out currentY); | ||
69 | |||
70 | List<GridRegion> candidateRegions = new List<GridRegion>(); | ||
71 | TryAddRegion(Utils.UIntsToLong(Math.Max(0, currentX - Constants.RegionSize), currentY), candidateRegions); // West | ||
72 | TryAddRegion(Utils.UIntsToLong(currentX + Constants.RegionSize, currentY), candidateRegions); // East | ||
73 | TryAddRegion(Utils.UIntsToLong(currentX, Math.Max(0, currentY - Constants.RegionSize)), candidateRegions); // South | ||
74 | TryAddRegion(Utils.UIntsToLong(currentX, currentY + Constants.RegionSize), candidateRegions); // North | ||
75 | |||
76 | if (candidateRegions.Count != 0) | ||
77 | { | ||
78 | GridRegion destRegion = candidateRegions[Bot.Manager.Rng.Next(candidateRegions.Count)]; | ||
79 | |||
80 | uint targetX, targetY; | ||
81 | Utils.LongToUInts(destRegion.RegionHandle, out targetX, out targetY); | ||
82 | |||
83 | Vector3 pos = client.Self.SimPosition; | ||
84 | if (targetX < currentX) | ||
85 | pos.X = -1; | ||
86 | else if (targetX > currentX) | ||
87 | pos.X = Constants.RegionSize + 1; | ||
88 | |||
89 | if (targetY < currentY) | ||
90 | pos.Y = -1; | ||
91 | else if (targetY > currentY) | ||
92 | pos.Y = Constants.RegionSize + 1; | ||
93 | |||
94 | m_log.DebugFormat( | ||
95 | "[CROSS BEHAVIOUR]: {0} moving to cross from {1} into {2}, target {3}", | ||
96 | Bot.Name, currentSim.Name, destRegion.Name, pos); | ||
97 | |||
98 | // Face in the direction of the candidate region | ||
99 | client.Self.Movement.TurnToward(pos); | ||
100 | |||
101 | // Listen for event so that we know when we've crossed the region boundary | ||
102 | Bot.Client.Self.RegionCrossed += Self_RegionCrossed; | ||
103 | |||
104 | // Start moving | ||
105 | Bot.Client.Self.Movement.AtPos = true; | ||
106 | |||
107 | // Stop when reach region target or border cross detected | ||
108 | if (!m_regionCrossedMutex.WaitOne(m_regionCrossingTimeout)) | ||
109 | { | ||
110 | m_log.ErrorFormat( | ||
111 | "[CROSS BEHAVIOUR]: {0} failed to cross from {1} into {2} with {3}ms", | ||
112 | Bot.Name, currentSim.Name, destRegion.Name, m_regionCrossingTimeout); | ||
113 | } | ||
114 | else | ||
115 | { | ||
116 | m_log.DebugFormat( | ||
117 | "[CROSS BEHAVIOUR]: {0} crossed from {1} into {2}", | ||
118 | Bot.Name, currentSim.Name, destRegion.Name); | ||
119 | } | ||
120 | |||
121 | Bot.Client.Self.RegionCrossed -= Self_RegionCrossed; | ||
122 | |||
123 | // We will hackishly carry on travelling into the region for a little bit. | ||
124 | Thread.Sleep(6000); | ||
125 | |||
126 | m_log.DebugFormat( | ||
127 | "[CROSS BEHAVIOUR]: {0} stopped moving after cross from {1} into {2}", | ||
128 | Bot.Name, currentSim.Name, destRegion.Name); | ||
129 | |||
130 | Bot.Client.Self.Movement.AtPos = false; | ||
131 | } | ||
132 | else | ||
133 | { | ||
134 | m_log.DebugFormat( | ||
135 | "[CROSS BEHAVIOUR]: No candidate region for {0} to cross into from {1}. Ignoring.", | ||
136 | Bot.Name, currentSim.Name); | ||
137 | } | ||
138 | } | ||
139 | |||
140 | private bool TryAddRegion(ulong handle, List<GridRegion> regions) | ||
141 | { | ||
142 | Dictionary<ulong, GridRegion> knownRegions = Bot.Manager.RegionsKnown; | ||
143 | |||
144 | lock (knownRegions) | ||
145 | { | ||
146 | if (knownRegions.Count == 0) | ||
147 | return false; | ||
148 | |||
149 | m_log.DebugFormat("[CROSS BEHAVIOUR]: Looking for region with handle {0} in known regions", handle); | ||
150 | |||
151 | if (knownRegions.ContainsKey(handle)) | ||
152 | { | ||
153 | GridRegion region = knownRegions[handle]; | ||
154 | m_log.DebugFormat( | ||
155 | "[CROSS BEHAVIOUR]: Adding region {0} to crossing candidates for {1}", region.Name, Bot.Name); | ||
156 | |||
157 | regions.Add(region); | ||
158 | |||
159 | return true; | ||
160 | } | ||
161 | else | ||
162 | { | ||
163 | return false; | ||
164 | } | ||
165 | } | ||
166 | } | ||
167 | |||
168 | internal void Self_RegionCrossed(object o, RegionCrossedEventArgs args) | ||
169 | { | ||
170 | m_regionCrossedMutex.Set(); | ||
171 | } | ||
172 | } | ||
173 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/pCampBot/Behaviours/GrabbingBehaviour.cs b/OpenSim/Tools/pCampBot/Behaviours/GrabbingBehaviour.cs new file mode 100644 index 0000000..59f6244 --- /dev/null +++ b/OpenSim/Tools/pCampBot/Behaviours/GrabbingBehaviour.cs | |||
@@ -0,0 +1,68 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using OpenMetaverse; | ||
29 | using System; | ||
30 | using System.Collections.Generic; | ||
31 | using System.Linq; | ||
32 | using System.Threading; | ||
33 | using pCampBot.Interfaces; | ||
34 | |||
35 | namespace pCampBot | ||
36 | { | ||
37 | /// <summary> | ||
38 | /// Click (grab) on random objects in the scene. | ||
39 | /// </summary> | ||
40 | /// <remarks> | ||
41 | /// The viewer itself does not give the option of grabbing objects that haven't been signalled as grabbable. | ||
42 | /// </remarks> | ||
43 | public class GrabbingBehaviour : AbstractBehaviour | ||
44 | { | ||
45 | public GrabbingBehaviour() | ||
46 | { | ||
47 | AbbreviatedName = "g"; | ||
48 | Name = "Grabbing"; | ||
49 | } | ||
50 | |||
51 | public override void Action() | ||
52 | { | ||
53 | Dictionary<UUID, Primitive> objects = Bot.Objects; | ||
54 | |||
55 | if (objects.Count <= 0) | ||
56 | return; | ||
57 | |||
58 | Primitive prim = objects.ElementAt(Bot.Random.Next(0, objects.Count - 1)).Value; | ||
59 | |||
60 | // This appears to be a typical message sent when a viewer user clicks a clickable object | ||
61 | Bot.Client.Self.Grab(prim.LocalID); | ||
62 | Bot.Client.Self.GrabUpdate(prim.ID, Vector3.Zero); | ||
63 | Bot.Client.Self.DeGrab(prim.LocalID); | ||
64 | |||
65 | Thread.Sleep(1000); | ||
66 | } | ||
67 | } | ||
68 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/pCampBot/Behaviours/InventoryDownloadBehaviour.cs b/OpenSim/Tools/pCampBot/Behaviours/InventoryDownloadBehaviour.cs new file mode 100644 index 0000000..521415c --- /dev/null +++ b/OpenSim/Tools/pCampBot/Behaviours/InventoryDownloadBehaviour.cs | |||
@@ -0,0 +1,121 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using OpenMetaverse; | ||
29 | using System; | ||
30 | using System.Collections.Generic; | ||
31 | using System.Diagnostics; | ||
32 | using System.Threading; | ||
33 | using System.Linq; | ||
34 | using pCampBot.Interfaces; | ||
35 | |||
36 | namespace pCampBot | ||
37 | { | ||
38 | /// <summary> | ||
39 | /// Do nothing | ||
40 | /// </summary> | ||
41 | public class InventoryDownloadBehaviour : AbstractBehaviour | ||
42 | { | ||
43 | private bool m_initialized; | ||
44 | private int m_Requests = 2; | ||
45 | private Stopwatch m_StopWatch = new Stopwatch(); | ||
46 | private List<UUID> m_processed = new List<UUID>(); | ||
47 | |||
48 | public InventoryDownloadBehaviour() | ||
49 | { | ||
50 | AbbreviatedName = "inv"; | ||
51 | Name = "Inventory"; | ||
52 | } | ||
53 | |||
54 | public override void Action() | ||
55 | { | ||
56 | if (!m_initialized) | ||
57 | { | ||
58 | m_initialized = true; | ||
59 | Bot.Client.Settings.HTTP_INVENTORY = true; | ||
60 | Bot.Client.Settings.FETCH_MISSING_INVENTORY = true; | ||
61 | Bot.Client.Inventory.FolderUpdated += Inventory_FolderUpdated; | ||
62 | Console.WriteLine("Lib owner is " + Bot.Client.Inventory.Store.LibraryRootNode.Data.OwnerID); | ||
63 | m_StopWatch.Start(); | ||
64 | Bot.Client.Inventory.RequestFolderContents(Bot.Client.Inventory.Store.RootFolder.UUID, Bot.Client.Self.AgentID, true, true, InventorySortOrder.ByDate); | ||
65 | Bot.Client.Inventory.RequestFolderContents(Bot.Client.Inventory.Store.LibraryRootNode.Data.UUID, Bot.Client.Inventory.Store.LibraryRootNode.Data.OwnerID, true, true, InventorySortOrder.ByDate); | ||
66 | } | ||
67 | |||
68 | Thread.Sleep(1000); | ||
69 | Console.WriteLine("Total items: " + Bot.Client.Inventory.Store.Items.Count + "; Total requests: " + m_Requests + "; Time: " + m_StopWatch.Elapsed); | ||
70 | |||
71 | } | ||
72 | |||
73 | void Inventory_FolderUpdated(object sender, FolderUpdatedEventArgs e) | ||
74 | { | ||
75 | if (e.Success) | ||
76 | { | ||
77 | //Console.WriteLine("Folder " + e.FolderID + " updated"); | ||
78 | bool fetch = false; | ||
79 | lock (m_processed) | ||
80 | { | ||
81 | if (!m_processed.Contains(e.FolderID)) | ||
82 | { | ||
83 | m_processed.Add(e.FolderID); | ||
84 | fetch = true; | ||
85 | } | ||
86 | } | ||
87 | |||
88 | if (fetch) | ||
89 | { | ||
90 | List<InventoryFolder> m_foldersToFetch = new List<InventoryFolder>(); | ||
91 | foreach (InventoryBase item in Bot.Client.Inventory.Store.GetContents(e.FolderID)) | ||
92 | { | ||
93 | if (item is InventoryFolder) | ||
94 | { | ||
95 | InventoryFolder f = new InventoryFolder(item.UUID); | ||
96 | f.OwnerID = item.OwnerID; | ||
97 | m_foldersToFetch.Add(f); | ||
98 | } | ||
99 | } | ||
100 | if (m_foldersToFetch.Count > 0) | ||
101 | { | ||
102 | m_Requests += 1; | ||
103 | Bot.Client.Inventory.RequestFolderContentsCap(m_foldersToFetch, Bot.Client.Network.CurrentSim.Caps.CapabilityURI("FetchInventoryDescendents2"), true, true, InventorySortOrder.ByDate); | ||
104 | } | ||
105 | } | ||
106 | |||
107 | if (Bot.Client.Inventory.Store.Items.Count >= 15739) | ||
108 | { | ||
109 | m_StopWatch.Stop(); | ||
110 | Console.WriteLine("Stop! Total items: " + Bot.Client.Inventory.Store.Items.Count + "; Total requests: " + m_Requests + "; Time: " + m_StopWatch.Elapsed); | ||
111 | } | ||
112 | } | ||
113 | |||
114 | } | ||
115 | |||
116 | public override void Interrupt() | ||
117 | { | ||
118 | m_interruptEvent.Set(); | ||
119 | } | ||
120 | } | ||
121 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/pCampBot/Behaviours/NoneBehaviour.cs b/OpenSim/Tools/pCampBot/Behaviours/NoneBehaviour.cs new file mode 100644 index 0000000..0d43781 --- /dev/null +++ b/OpenSim/Tools/pCampBot/Behaviours/NoneBehaviour.cs | |||
@@ -0,0 +1,60 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using OpenMetaverse; | ||
29 | using System; | ||
30 | using System.Collections.Generic; | ||
31 | using System.Linq; | ||
32 | using pCampBot.Interfaces; | ||
33 | |||
34 | namespace pCampBot | ||
35 | { | ||
36 | /// <summary> | ||
37 | /// Do nothing | ||
38 | /// </summary> | ||
39 | public class NoneBehaviour : AbstractBehaviour | ||
40 | { | ||
41 | public NoneBehaviour() | ||
42 | { | ||
43 | AbbreviatedName = "n"; | ||
44 | Name = "None"; | ||
45 | } | ||
46 | |||
47 | public override void Action() | ||
48 | { | ||
49 | Bot.Client.Self.Jump(false); | ||
50 | Bot.Client.Self.Movement.Stop = true; | ||
51 | m_interruptEvent.WaitOne(); | ||
52 | Bot.Client.Self.Movement.Stop = false; | ||
53 | } | ||
54 | |||
55 | public override void Interrupt() | ||
56 | { | ||
57 | m_interruptEvent.Set(); | ||
58 | } | ||
59 | } | ||
60 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/pCampBot/Behaviours/PhysicsBehaviour.cs b/OpenSim/Tools/pCampBot/Behaviours/PhysicsBehaviour.cs new file mode 100644 index 0000000..98ab931 --- /dev/null +++ b/OpenSim/Tools/pCampBot/Behaviours/PhysicsBehaviour.cs | |||
@@ -0,0 +1,104 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.IO; | ||
30 | using System.Threading; | ||
31 | using OpenMetaverse; | ||
32 | using OpenSim.Framework; | ||
33 | using pCampBot.Interfaces; | ||
34 | |||
35 | namespace pCampBot | ||
36 | { | ||
37 | /// <summary> | ||
38 | /// Stress physics by moving and bouncing around bots a whole lot. | ||
39 | /// </summary> | ||
40 | /// <remarks> | ||
41 | /// TODO: talkarray should be in a separate behaviour. | ||
42 | /// </remarks> | ||
43 | public class PhysicsBehaviour : AbstractBehaviour | ||
44 | { | ||
45 | private string[] talkarray; | ||
46 | |||
47 | public PhysicsBehaviour() | ||
48 | { | ||
49 | AbbreviatedName = "p"; | ||
50 | Name = "Physics"; | ||
51 | talkarray = readexcuses(); | ||
52 | } | ||
53 | |||
54 | public override void Action() | ||
55 | { | ||
56 | int walkorrun = Bot.Random.Next(4); // Randomize between walking and running. The greater this number, | ||
57 | // the greater the bot's chances to walk instead of run. | ||
58 | Bot.Client.Self.Jump(false); | ||
59 | if (walkorrun == 0) | ||
60 | { | ||
61 | Bot.Client.Self.Movement.AlwaysRun = true; | ||
62 | } | ||
63 | else | ||
64 | { | ||
65 | Bot.Client.Self.Movement.AlwaysRun = false; | ||
66 | } | ||
67 | |||
68 | // TODO: unused: Vector3 pos = client.Self.SimPosition; | ||
69 | Vector3 newpos = new Vector3(Bot.Random.Next(1, 254), Bot.Random.Next(1, 254), Bot.Random.Next(1, 254)); | ||
70 | Bot.Client.Self.Movement.TurnToward(newpos); | ||
71 | |||
72 | Bot.Client.Self.Movement.AtPos = true; | ||
73 | Thread.Sleep(Bot.Random.Next(3000, 13000)); | ||
74 | Bot.Client.Self.Movement.AtPos = false; | ||
75 | Bot.Client.Self.Jump(true); | ||
76 | string randomf = talkarray[Bot.Random.Next(talkarray.Length)]; | ||
77 | if (talkarray.Length > 1 && randomf.Length > 1) | ||
78 | Bot.Client.Self.Chat(randomf, 0, ChatType.Normal); | ||
79 | } | ||
80 | |||
81 | public override void Close() | ||
82 | { | ||
83 | if (Bot.ConnectionState == ConnectionState.Connected) | ||
84 | Bot.Client.Self.Jump(false); | ||
85 | |||
86 | base.Close(); | ||
87 | } | ||
88 | |||
89 | private string[] readexcuses() | ||
90 | { | ||
91 | string allexcuses = ""; | ||
92 | |||
93 | string file = Path.Combine(Util.configDir(), "pCampBotSentences.txt"); | ||
94 | if (File.Exists(file)) | ||
95 | { | ||
96 | StreamReader csr = File.OpenText(file); | ||
97 | allexcuses = csr.ReadToEnd(); | ||
98 | csr.Close(); | ||
99 | } | ||
100 | |||
101 | return allexcuses.Split(Environment.NewLine.ToCharArray()); | ||
102 | } | ||
103 | } | ||
104 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/pCampBot/Behaviours/PhysicsBehaviour2.cs b/OpenSim/Tools/pCampBot/Behaviours/PhysicsBehaviour2.cs new file mode 100644 index 0000000..1ec2046 --- /dev/null +++ b/OpenSim/Tools/pCampBot/Behaviours/PhysicsBehaviour2.cs | |||
@@ -0,0 +1,86 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using OpenMetaverse; | ||
29 | using System; | ||
30 | using System.Collections.Generic; | ||
31 | using System.Linq; | ||
32 | using System.Reflection; | ||
33 | using System.Threading; | ||
34 | using log4net; | ||
35 | using pCampBot.Interfaces; | ||
36 | |||
37 | namespace pCampBot | ||
38 | { | ||
39 | /// <summary> | ||
40 | /// This behavior is for the systematic study of some performance improvements made | ||
41 | /// for OSCC'13. | ||
42 | /// Walk around, sending AgentUpdate packets all the time. | ||
43 | /// </summary> | ||
44 | public class PhysicsBehaviour2 : AbstractBehaviour | ||
45 | { | ||
46 | // private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); | ||
47 | |||
48 | public PhysicsBehaviour2() | ||
49 | { | ||
50 | AbbreviatedName = "ph2"; | ||
51 | Name = "Physics2"; | ||
52 | } | ||
53 | |||
54 | private const int TIME_WALKING = 5 * 10; // 5 seconds | ||
55 | private int counter = 0; | ||
56 | |||
57 | public override void Action() | ||
58 | { | ||
59 | |||
60 | if (counter >= TIME_WALKING) | ||
61 | { | ||
62 | counter = 0; | ||
63 | |||
64 | Vector3 target = new Vector3(Bot.Random.Next(1, 254), Bot.Random.Next(1, 254), Bot.Client.Self.SimPosition.Z); | ||
65 | MyTurnToward(target); | ||
66 | |||
67 | Bot.Client.Self.Movement.AtPos = true; | ||
68 | |||
69 | } | ||
70 | else | ||
71 | counter++; | ||
72 | // In any case, send an update | ||
73 | Bot.Client.Self.Movement.SendUpdate(); | ||
74 | } | ||
75 | |||
76 | private void MyTurnToward(Vector3 target) | ||
77 | { | ||
78 | Quaternion between = Vector3.RotationBetween(Vector3.UnitX, Vector3.Normalize(target - Bot.Client.Self.SimPosition)); | ||
79 | Quaternion rot = between ; | ||
80 | |||
81 | Bot.Client.Self.Movement.BodyRotation = rot; | ||
82 | Bot.Client.Self.Movement.HeadRotation = rot; | ||
83 | Bot.Client.Self.Movement.Camera.LookAt(Bot.Client.Self.SimPosition, target); | ||
84 | } | ||
85 | } | ||
86 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/pCampBot/Behaviours/TeleportBehaviour.cs b/OpenSim/Tools/pCampBot/Behaviours/TeleportBehaviour.cs new file mode 100644 index 0000000..81f250d --- /dev/null +++ b/OpenSim/Tools/pCampBot/Behaviours/TeleportBehaviour.cs | |||
@@ -0,0 +1,82 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.Collections.Generic; | ||
30 | using System.Linq; | ||
31 | using System.Reflection; | ||
32 | using System.Threading; | ||
33 | using log4net; | ||
34 | using OpenMetaverse; | ||
35 | using pCampBot.Interfaces; | ||
36 | |||
37 | namespace pCampBot | ||
38 | { | ||
39 | /// <summary> | ||
40 | /// Teleport to a random region on the grid. | ||
41 | /// </summary> | ||
42 | public class TeleportBehaviour : AbstractBehaviour | ||
43 | { | ||
44 | private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); | ||
45 | |||
46 | public TeleportBehaviour() | ||
47 | { | ||
48 | AbbreviatedName = "t"; | ||
49 | Name = "Teleport"; | ||
50 | } | ||
51 | |||
52 | public override void Action() | ||
53 | { | ||
54 | Random rng = Bot.Manager.Rng; | ||
55 | GridRegion[] knownRegions; | ||
56 | |||
57 | lock (Bot.Manager.RegionsKnown) | ||
58 | { | ||
59 | if (Bot.Manager.RegionsKnown.Count == 0) | ||
60 | { | ||
61 | m_log.DebugFormat( | ||
62 | "[TELEPORT BEHAVIOUR]: Ignoring teleport action for {0} since no regions are known yet", Bot.Name); | ||
63 | return; | ||
64 | } | ||
65 | |||
66 | knownRegions = Bot.Manager.RegionsKnown.Values.ToArray(); | ||
67 | } | ||
68 | |||
69 | Simulator sourceRegion = Bot.Client.Network.CurrentSim; | ||
70 | GridRegion destRegion = knownRegions[rng.Next(knownRegions.Length)]; | ||
71 | Vector3 destPosition = new Vector3(rng.Next(255), rng.Next(255), 50); | ||
72 | |||
73 | m_log.DebugFormat( | ||
74 | "[TELEPORT BEHAVIOUR]: Teleporting {0} from {1} {2} to {3} {4}", | ||
75 | Bot.Name, sourceRegion.Name, Bot.Client.Self.SimPosition, destRegion.Name, destPosition); | ||
76 | |||
77 | Bot.Client.Self.Teleport(destRegion.RegionHandle, destPosition); | ||
78 | |||
79 | Thread.Sleep(Bot.Random.Next(3000, 10000)); | ||
80 | } | ||
81 | } | ||
82 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/pCampBot/Behaviours/TwitchyBehaviour.cs b/OpenSim/Tools/pCampBot/Behaviours/TwitchyBehaviour.cs new file mode 100644 index 0000000..7b4639d --- /dev/null +++ b/OpenSim/Tools/pCampBot/Behaviours/TwitchyBehaviour.cs | |||
@@ -0,0 +1,73 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using OpenMetaverse; | ||
29 | using System; | ||
30 | using System.Collections.Generic; | ||
31 | using System.Linq; | ||
32 | using System.Reflection; | ||
33 | using log4net; | ||
34 | using pCampBot.Interfaces; | ||
35 | |||
36 | namespace pCampBot | ||
37 | { | ||
38 | /// <summary> | ||
39 | /// This behavior is for the systematic study of some performance improvements made | ||
40 | /// for OSCC'13. | ||
41 | /// Do nothing, but send AgentUpdate packets all the time that have only slightly | ||
42 | /// different state. The delta of difference will be filtered by OpenSim early on | ||
43 | /// in the packet processing pipeline. These filters did not exist before OSCC'13. | ||
44 | /// </summary> | ||
45 | public class TwitchyBehaviour : AbstractBehaviour | ||
46 | { | ||
47 | // private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); | ||
48 | |||
49 | public TwitchyBehaviour() | ||
50 | { | ||
51 | AbbreviatedName = "tw"; | ||
52 | Name = "Twitchy"; | ||
53 | } | ||
54 | |||
55 | private const float TWITCH = 0.0001f; | ||
56 | private int direction = 1; | ||
57 | |||
58 | public override void Action() | ||
59 | { | ||
60 | Bot.Client.Self.Movement.BodyRotation = new Quaternion(Bot.Client.Self.Movement.BodyRotation.X + direction * TWITCH, | ||
61 | Bot.Client.Self.Movement.BodyRotation.Y, | ||
62 | Bot.Client.Self.Movement.BodyRotation.Z, | ||
63 | Bot.Client.Self.Movement.BodyRotation.W); | ||
64 | |||
65 | //m_log.DebugFormat("[TWITCH]: BodyRot {0}", Bot.Client.Self.Movement.BodyRotation); | ||
66 | direction = -direction; | ||
67 | |||
68 | Bot.Client.Self.Movement.SendUpdate(); | ||
69 | |||
70 | } | ||
71 | |||
72 | } | ||
73 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/pCampBot/Bot.cs b/OpenSim/Tools/pCampBot/Bot.cs new file mode 100644 index 0000000..4f28733 --- /dev/null +++ b/OpenSim/Tools/pCampBot/Bot.cs | |||
@@ -0,0 +1,748 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.Collections.Generic; | ||
30 | using System.Text; | ||
31 | using System.IO; | ||
32 | using System.Reflection; | ||
33 | using System.Threading; | ||
34 | using System.Timers; | ||
35 | using log4net; | ||
36 | using OpenMetaverse; | ||
37 | using OpenMetaverse.Assets; | ||
38 | using OpenMetaverse.Packets; | ||
39 | using Nini.Config; | ||
40 | using OpenSim.Framework; | ||
41 | using OpenSim.Framework.Console; | ||
42 | using pCampBot.Interfaces; | ||
43 | using Timer = System.Timers.Timer; | ||
44 | using PermissionMask = OpenSim.Framework.PermissionMask; | ||
45 | |||
46 | namespace pCampBot | ||
47 | { | ||
48 | public enum ConnectionState | ||
49 | { | ||
50 | Disconnected, | ||
51 | Connecting, | ||
52 | Connected, | ||
53 | Disconnecting | ||
54 | } | ||
55 | |||
56 | public class Bot | ||
57 | { | ||
58 | private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); | ||
59 | |||
60 | public int PacketDebugLevel | ||
61 | { | ||
62 | get { return m_packetDebugLevel; } | ||
63 | set | ||
64 | { | ||
65 | if (value == m_packetDebugLevel) | ||
66 | return; | ||
67 | |||
68 | m_packetDebugLevel = value; | ||
69 | |||
70 | if (Client != null) | ||
71 | { | ||
72 | if (m_packetDebugLevel <= 0) | ||
73 | Client.Network.UnregisterCallback(PacketType.Default, PacketReceivedDebugHandler); | ||
74 | else | ||
75 | Client.Network.RegisterCallback(PacketType.Default, PacketReceivedDebugHandler, false); | ||
76 | } | ||
77 | } | ||
78 | } | ||
79 | private int m_packetDebugLevel; | ||
80 | |||
81 | public delegate void AnEvent(Bot callbot, EventType someevent); // event delegate for bot events | ||
82 | |||
83 | /// <summary> | ||
84 | /// Controls whether bots request textures for the object information they receive | ||
85 | /// </summary> | ||
86 | public bool RequestObjectTextures { get; set; } | ||
87 | |||
88 | /// <summary> | ||
89 | /// Bot manager. | ||
90 | /// </summary> | ||
91 | public BotManager Manager { get; private set; } | ||
92 | |||
93 | /// <summary> | ||
94 | /// Behaviours implemented by this bot. | ||
95 | /// </summary> | ||
96 | /// <remarks> | ||
97 | /// Indexed by abbreviated name. There can only be one instance of a particular behaviour. | ||
98 | /// Lock this structure before manipulating it. | ||
99 | /// </remarks> | ||
100 | public Dictionary<string, IBehaviour> Behaviours { get; private set; } | ||
101 | |||
102 | /// <summary> | ||
103 | /// Objects that the bot has discovered. | ||
104 | /// </summary> | ||
105 | /// <remarks> | ||
106 | /// Returns a list copy. Inserting new objects manually will have no effect. | ||
107 | /// </remarks> | ||
108 | public Dictionary<UUID, Primitive> Objects | ||
109 | { | ||
110 | get | ||
111 | { | ||
112 | lock (m_objects) | ||
113 | return new Dictionary<UUID, Primitive>(m_objects); | ||
114 | } | ||
115 | } | ||
116 | private Dictionary<UUID, Primitive> m_objects = new Dictionary<UUID, Primitive>(); | ||
117 | |||
118 | /// <summary> | ||
119 | /// Is this bot connected to the grid? | ||
120 | /// </summary> | ||
121 | public ConnectionState ConnectionState { get; private set; } | ||
122 | |||
123 | public List<Simulator> Simulators | ||
124 | { | ||
125 | get | ||
126 | { | ||
127 | lock (Client.Network.Simulators) | ||
128 | return new List<Simulator>(Client.Network.Simulators); | ||
129 | } | ||
130 | } | ||
131 | |||
132 | /// <summary> | ||
133 | /// The number of connections that this bot has to different simulators. | ||
134 | /// </summary> | ||
135 | /// <value>Includes both root and child connections.</value> | ||
136 | public int SimulatorsCount | ||
137 | { | ||
138 | get | ||
139 | { | ||
140 | lock (Client.Network.Simulators) | ||
141 | return Client.Network.Simulators.Count; | ||
142 | } | ||
143 | } | ||
144 | |||
145 | public string FirstName { get; private set; } | ||
146 | public string LastName { get; private set; } | ||
147 | public string Name { get; private set; } | ||
148 | public string Password { get; private set; } | ||
149 | public string LoginUri { get; private set; } | ||
150 | public string StartLocation { get; private set; } | ||
151 | |||
152 | public string saveDir; | ||
153 | public string wear; | ||
154 | |||
155 | public event AnEvent OnConnected; | ||
156 | public event AnEvent OnDisconnected; | ||
157 | |||
158 | /// <summary> | ||
159 | /// Keep a track of the continuously acting thread so that we can abort it. | ||
160 | /// </summary> | ||
161 | private Thread m_actionThread; | ||
162 | |||
163 | protected List<uint> objectIDs = new List<uint>(); | ||
164 | |||
165 | /// <summary> | ||
166 | /// Random number generator. | ||
167 | /// </summary> | ||
168 | public Random Random { get; private set; } | ||
169 | |||
170 | /// <summary> | ||
171 | /// New instance of a SecondLife client | ||
172 | /// </summary> | ||
173 | public GridClient Client { get; private set; } | ||
174 | |||
175 | /// <summary> | ||
176 | /// Constructor | ||
177 | /// </summary> | ||
178 | /// <param name="bm"></param> | ||
179 | /// <param name="behaviours">Behaviours for this bot to perform</param> | ||
180 | /// <param name="firstName"></param> | ||
181 | /// <param name="lastName"></param> | ||
182 | /// <param name="password"></param> | ||
183 | /// <param name="loginUri"></param> | ||
184 | /// <param name="behaviours"></param> | ||
185 | public Bot( | ||
186 | BotManager bm, List<IBehaviour> behaviours, | ||
187 | string firstName, string lastName, string password, string startLocation, string loginUri) | ||
188 | { | ||
189 | ConnectionState = ConnectionState.Disconnected; | ||
190 | |||
191 | Random = new Random(bm.Rng.Next()); | ||
192 | FirstName = firstName; | ||
193 | LastName = lastName; | ||
194 | Name = string.Format("{0} {1}", FirstName, LastName); | ||
195 | Password = password; | ||
196 | LoginUri = loginUri; | ||
197 | StartLocation = startLocation; | ||
198 | |||
199 | Manager = bm; | ||
200 | |||
201 | Behaviours = new Dictionary<string, IBehaviour>(); | ||
202 | foreach (IBehaviour behaviour in behaviours) | ||
203 | AddBehaviour(behaviour); | ||
204 | |||
205 | // Only calling for use as a template. | ||
206 | CreateLibOmvClient(); | ||
207 | } | ||
208 | |||
209 | public bool TryGetBehaviour(string abbreviatedName, out IBehaviour behaviour) | ||
210 | { | ||
211 | lock (Behaviours) | ||
212 | return Behaviours.TryGetValue(abbreviatedName, out behaviour); | ||
213 | } | ||
214 | |||
215 | public bool AddBehaviour(IBehaviour behaviour) | ||
216 | { | ||
217 | Dictionary<string, IBehaviour> updatedBehaviours = new Dictionary<string, IBehaviour>(Behaviours); | ||
218 | |||
219 | if (!updatedBehaviours.ContainsKey(behaviour.AbbreviatedName)) | ||
220 | { | ||
221 | behaviour.Initialize(this); | ||
222 | updatedBehaviours.Add(behaviour.AbbreviatedName, behaviour); | ||
223 | Behaviours = updatedBehaviours; | ||
224 | |||
225 | return true; | ||
226 | } | ||
227 | |||
228 | return false; | ||
229 | } | ||
230 | |||
231 | public bool RemoveBehaviour(string abbreviatedName) | ||
232 | { | ||
233 | if (Behaviours.Count <= 0) | ||
234 | return false; | ||
235 | |||
236 | Dictionary<string, IBehaviour> updatedBehaviours = new Dictionary<string, IBehaviour>(Behaviours); | ||
237 | IBehaviour behaviour; | ||
238 | |||
239 | if (!updatedBehaviours.TryGetValue(abbreviatedName, out behaviour)) | ||
240 | return false; | ||
241 | |||
242 | updatedBehaviours.Remove(abbreviatedName); | ||
243 | Behaviours = updatedBehaviours; | ||
244 | |||
245 | behaviour.Close(); | ||
246 | |||
247 | return true; | ||
248 | } | ||
249 | |||
250 | private void CreateLibOmvClient() | ||
251 | { | ||
252 | GridClient newClient = new GridClient(); | ||
253 | |||
254 | if (Client != null) | ||
255 | { | ||
256 | // Remove any registered debug handlers | ||
257 | Client.Network.UnregisterCallback(PacketType.Default, PacketReceivedDebugHandler); | ||
258 | |||
259 | newClient.Settings.LOGIN_SERVER = Client.Settings.LOGIN_SERVER; | ||
260 | newClient.Settings.ALWAYS_DECODE_OBJECTS = Client.Settings.ALWAYS_DECODE_OBJECTS; | ||
261 | newClient.Settings.AVATAR_TRACKING = Client.Settings.AVATAR_TRACKING; | ||
262 | newClient.Settings.OBJECT_TRACKING = Client.Settings.OBJECT_TRACKING; | ||
263 | newClient.Settings.SEND_AGENT_THROTTLE = Client.Settings.SEND_AGENT_THROTTLE; | ||
264 | newClient.Settings.SEND_AGENT_UPDATES = Client.Settings.SEND_AGENT_UPDATES; | ||
265 | newClient.Settings.SEND_PINGS = Client.Settings.SEND_PINGS; | ||
266 | newClient.Settings.STORE_LAND_PATCHES = Client.Settings.STORE_LAND_PATCHES; | ||
267 | newClient.Settings.USE_ASSET_CACHE = Client.Settings.USE_ASSET_CACHE; | ||
268 | newClient.Settings.MULTIPLE_SIMS = Client.Settings.MULTIPLE_SIMS; | ||
269 | newClient.Throttle.Asset = Client.Throttle.Asset; | ||
270 | newClient.Throttle.Land = Client.Throttle.Land; | ||
271 | newClient.Throttle.Task = Client.Throttle.Task; | ||
272 | newClient.Throttle.Texture = Client.Throttle.Texture; | ||
273 | newClient.Throttle.Wind = Client.Throttle.Wind; | ||
274 | newClient.Throttle.Total = Client.Throttle.Total; | ||
275 | } | ||
276 | else | ||
277 | { | ||
278 | newClient.Settings.LOGIN_SERVER = LoginUri; | ||
279 | newClient.Settings.ALWAYS_DECODE_OBJECTS = false; | ||
280 | newClient.Settings.AVATAR_TRACKING = false; | ||
281 | newClient.Settings.OBJECT_TRACKING = false; | ||
282 | newClient.Settings.SEND_AGENT_THROTTLE = true; | ||
283 | newClient.Settings.SEND_PINGS = true; | ||
284 | newClient.Settings.STORE_LAND_PATCHES = false; | ||
285 | newClient.Settings.USE_ASSET_CACHE = false; | ||
286 | newClient.Settings.MULTIPLE_SIMS = true; | ||
287 | newClient.Throttle.Asset = 100000; | ||
288 | newClient.Throttle.Land = 100000; | ||
289 | newClient.Throttle.Task = 100000; | ||
290 | newClient.Throttle.Texture = 100000; | ||
291 | newClient.Throttle.Wind = 100000; | ||
292 | newClient.Throttle.Total = 400000; | ||
293 | } | ||
294 | |||
295 | newClient.Network.LoginProgress += Network_LoginProgress; | ||
296 | newClient.Network.SimConnected += Network_SimConnected; | ||
297 | newClient.Network.SimDisconnected += Network_SimDisconnected; | ||
298 | newClient.Network.Disconnected += Network_OnDisconnected; | ||
299 | newClient.Objects.ObjectUpdate += Objects_NewPrim; | ||
300 | |||
301 | if (m_packetDebugLevel > 0) | ||
302 | newClient.Network.RegisterCallback(PacketType.Default, PacketReceivedDebugHandler); | ||
303 | |||
304 | Client = newClient; | ||
305 | } | ||
306 | |||
307 | //We do our actions here. This is where one would | ||
308 | //add additional steps and/or things the bot should do | ||
309 | private void Action() | ||
310 | { | ||
311 | while (ConnectionState == ConnectionState.Connected) | ||
312 | { | ||
313 | foreach (IBehaviour behaviour in Behaviours.Values) | ||
314 | { | ||
315 | // Thread.Sleep(Random.Next(3000, 10000)); | ||
316 | |||
317 | // m_log.DebugFormat("[pCAMPBOT]: For {0} performing action {1}", Name, b.GetType()); | ||
318 | behaviour.Action(); | ||
319 | } | ||
320 | } | ||
321 | |||
322 | foreach (IBehaviour b in Behaviours.Values) | ||
323 | b.Close(); | ||
324 | } | ||
325 | |||
326 | /// <summary> | ||
327 | /// Tells LibSecondLife to logout and disconnect. Raises the disconnect events once it finishes. | ||
328 | /// </summary> | ||
329 | public void Disconnect() | ||
330 | { | ||
331 | ConnectionState = ConnectionState.Disconnecting; | ||
332 | |||
333 | foreach (IBehaviour behaviour in Behaviours.Values) | ||
334 | behaviour.Close(); | ||
335 | |||
336 | Client.Network.Logout(); | ||
337 | } | ||
338 | |||
339 | public void Connect() | ||
340 | { | ||
341 | Thread connectThread = new Thread(ConnectInternal); | ||
342 | connectThread.Name = Name; | ||
343 | connectThread.IsBackground = true; | ||
344 | |||
345 | connectThread.Start(); | ||
346 | } | ||
347 | |||
348 | /// <summary> | ||
349 | /// This is the bot startup loop. | ||
350 | /// </summary> | ||
351 | private void ConnectInternal() | ||
352 | { | ||
353 | ConnectionState = ConnectionState.Connecting; | ||
354 | |||
355 | // Current create a new client on each connect. libomv doesn't seem to process new sim | ||
356 | // information (e.g. EstablishAgentCommunication events) if connecting after a disceonnect with the same | ||
357 | // client | ||
358 | CreateLibOmvClient(); | ||
359 | |||
360 | if (Client.Network.Login(FirstName, LastName, Password, "pCampBot", StartLocation, "pCampBot")) | ||
361 | { | ||
362 | ConnectionState = ConnectionState.Connected; | ||
363 | |||
364 | Thread.Sleep(Random.Next(1000, 10000)); | ||
365 | m_actionThread = new Thread(Action); | ||
366 | m_actionThread.Start(); | ||
367 | |||
368 | // OnConnected(this, EventType.CONNECTED); | ||
369 | if (wear == "save") | ||
370 | { | ||
371 | SaveDefaultAppearance(); | ||
372 | } | ||
373 | else if (wear != "no") | ||
374 | { | ||
375 | MakeDefaultAppearance(wear); | ||
376 | } | ||
377 | |||
378 | // Extract nearby region information. | ||
379 | Client.Grid.GridRegion += Manager.Grid_GridRegion; | ||
380 | uint xUint, yUint; | ||
381 | Utils.LongToUInts(Client.Network.CurrentSim.Handle, out xUint, out yUint); | ||
382 | ushort minX, minY, maxX, maxY; | ||
383 | minX = (ushort)Math.Min(0, xUint - 5); | ||
384 | minY = (ushort)Math.Min(0, yUint - 5); | ||
385 | maxX = (ushort)(xUint + 5); | ||
386 | maxY = (ushort)(yUint + 5); | ||
387 | Client.Grid.RequestMapBlocks(GridLayerType.Terrain, minX, minY, maxX, maxY, false); | ||
388 | } | ||
389 | else | ||
390 | { | ||
391 | ConnectionState = ConnectionState.Disconnected; | ||
392 | |||
393 | m_log.ErrorFormat( | ||
394 | "{0} {1} cannot login: {2}", FirstName, LastName, Client.Network.LoginMessage); | ||
395 | |||
396 | if (OnDisconnected != null) | ||
397 | { | ||
398 | OnDisconnected(this, EventType.DISCONNECTED); | ||
399 | } | ||
400 | } | ||
401 | } | ||
402 | |||
403 | /// <summary> | ||
404 | /// Sit this bot on the ground. | ||
405 | /// </summary> | ||
406 | public void SitOnGround() | ||
407 | { | ||
408 | if (ConnectionState == ConnectionState.Connected) | ||
409 | Client.Self.SitOnGround(); | ||
410 | } | ||
411 | |||
412 | /// <summary> | ||
413 | /// Stand this bot | ||
414 | /// </summary> | ||
415 | public void Stand() | ||
416 | { | ||
417 | if (ConnectionState == ConnectionState.Connected) | ||
418 | { | ||
419 | // Unlike sit on ground, here libomv checks whether we have SEND_AGENT_UPDATES enabled. | ||
420 | bool prevUpdatesSetting = Client.Settings.SEND_AGENT_UPDATES; | ||
421 | Client.Settings.SEND_AGENT_UPDATES = true; | ||
422 | Client.Self.Stand(); | ||
423 | Client.Settings.SEND_AGENT_UPDATES = prevUpdatesSetting; | ||
424 | } | ||
425 | } | ||
426 | |||
427 | public void SaveDefaultAppearance() | ||
428 | { | ||
429 | saveDir = "MyAppearance/" + FirstName + "_" + LastName; | ||
430 | if (!Directory.Exists(saveDir)) | ||
431 | { | ||
432 | Directory.CreateDirectory(saveDir); | ||
433 | } | ||
434 | |||
435 | Array wtypes = Enum.GetValues(typeof(WearableType)); | ||
436 | foreach (WearableType wtype in wtypes) | ||
437 | { | ||
438 | UUID wearable = Client.Appearance.GetWearableAsset(wtype); | ||
439 | if (wearable != UUID.Zero) | ||
440 | { | ||
441 | Client.Assets.RequestAsset(wearable, AssetType.Clothing, false, Asset_ReceivedCallback); | ||
442 | Client.Assets.RequestAsset(wearable, AssetType.Bodypart, false, Asset_ReceivedCallback); | ||
443 | } | ||
444 | } | ||
445 | } | ||
446 | |||
447 | public void SaveAsset(AssetWearable asset) | ||
448 | { | ||
449 | if (asset != null) | ||
450 | { | ||
451 | try | ||
452 | { | ||
453 | if (asset.Decode()) | ||
454 | { | ||
455 | File.WriteAllBytes(Path.Combine(saveDir, String.Format("{1}.{0}", | ||
456 | asset.AssetType.ToString().ToLower(), | ||
457 | asset.WearableType)), asset.AssetData); | ||
458 | } | ||
459 | else | ||
460 | { | ||
461 | m_log.WarnFormat("Failed to decode {0} asset {1}", asset.AssetType, asset.AssetID); | ||
462 | } | ||
463 | } | ||
464 | catch (Exception e) | ||
465 | { | ||
466 | m_log.ErrorFormat("Exception: {0}{1}", e.Message, e.StackTrace); | ||
467 | } | ||
468 | } | ||
469 | } | ||
470 | |||
471 | public WearableType GetWearableType(string path) | ||
472 | { | ||
473 | string type = ((((path.Split('/'))[2]).Split('.'))[0]).Trim(); | ||
474 | switch (type) | ||
475 | { | ||
476 | case "Eyes": | ||
477 | return WearableType.Eyes; | ||
478 | case "Hair": | ||
479 | return WearableType.Hair; | ||
480 | case "Pants": | ||
481 | return WearableType.Pants; | ||
482 | case "Shape": | ||
483 | return WearableType.Shape; | ||
484 | case "Shirt": | ||
485 | return WearableType.Shirt; | ||
486 | case "Skin": | ||
487 | return WearableType.Skin; | ||
488 | default: | ||
489 | return WearableType.Shape; | ||
490 | } | ||
491 | } | ||
492 | |||
493 | public void MakeDefaultAppearance(string wear) | ||
494 | { | ||
495 | try | ||
496 | { | ||
497 | if (wear == "yes") | ||
498 | { | ||
499 | //TODO: Implement random outfit picking | ||
500 | m_log.DebugFormat("Picks a random outfit. Not yet implemented."); | ||
501 | } | ||
502 | else if (wear != "save") | ||
503 | saveDir = "MyAppearance/" + wear; | ||
504 | saveDir = saveDir + "/"; | ||
505 | |||
506 | string[] clothing = Directory.GetFiles(saveDir, "*.clothing", SearchOption.TopDirectoryOnly); | ||
507 | string[] bodyparts = Directory.GetFiles(saveDir, "*.bodypart", SearchOption.TopDirectoryOnly); | ||
508 | InventoryFolder clothfolder = FindClothingFolder(); | ||
509 | UUID transid = UUID.Random(); | ||
510 | List<InventoryBase> listwearables = new List<InventoryBase>(); | ||
511 | |||
512 | for (int i = 0; i < clothing.Length; i++) | ||
513 | { | ||
514 | UUID assetID = UUID.Random(); | ||
515 | AssetClothing asset = new AssetClothing(assetID, File.ReadAllBytes(clothing[i])); | ||
516 | asset.Decode(); | ||
517 | asset.Owner = Client.Self.AgentID; | ||
518 | asset.WearableType = GetWearableType(clothing[i]); | ||
519 | asset.Encode(); | ||
520 | transid = Client.Assets.RequestUpload(asset,true); | ||
521 | Client.Inventory.RequestCreateItem(clothfolder.UUID, "MyClothing" + i.ToString(), "MyClothing", AssetType.Clothing, | ||
522 | transid, InventoryType.Wearable, asset.WearableType, (OpenMetaverse.PermissionMask)PermissionMask.All, delegate(bool success, InventoryItem item) | ||
523 | { | ||
524 | if (success) | ||
525 | { | ||
526 | listwearables.Add(item); | ||
527 | } | ||
528 | else | ||
529 | { | ||
530 | m_log.WarnFormat("Failed to create item {0}", item.Name); | ||
531 | } | ||
532 | } | ||
533 | ); | ||
534 | } | ||
535 | |||
536 | for (int i = 0; i < bodyparts.Length; i++) | ||
537 | { | ||
538 | UUID assetID = UUID.Random(); | ||
539 | AssetBodypart asset = new AssetBodypart(assetID, File.ReadAllBytes(bodyparts[i])); | ||
540 | asset.Decode(); | ||
541 | asset.Owner = Client.Self.AgentID; | ||
542 | asset.WearableType = GetWearableType(bodyparts[i]); | ||
543 | asset.Encode(); | ||
544 | transid = Client.Assets.RequestUpload(asset,true); | ||
545 | Client.Inventory.RequestCreateItem(clothfolder.UUID, "MyBodyPart" + i.ToString(), "MyBodyPart", AssetType.Bodypart, | ||
546 | transid, InventoryType.Wearable, asset.WearableType, (OpenMetaverse.PermissionMask)PermissionMask.All, delegate(bool success, InventoryItem item) | ||
547 | { | ||
548 | if (success) | ||
549 | { | ||
550 | listwearables.Add(item); | ||
551 | } | ||
552 | else | ||
553 | { | ||
554 | m_log.WarnFormat("Failed to create item {0}", item.Name); | ||
555 | } | ||
556 | } | ||
557 | ); | ||
558 | } | ||
559 | |||
560 | Thread.Sleep(1000); | ||
561 | |||
562 | if (listwearables == null || listwearables.Count == 0) | ||
563 | { | ||
564 | m_log.DebugFormat("Nothing to send on this folder!"); | ||
565 | } | ||
566 | else | ||
567 | { | ||
568 | m_log.DebugFormat("Sending {0} wearables...", listwearables.Count); | ||
569 | Client.Appearance.WearOutfit(listwearables, false); | ||
570 | } | ||
571 | } | ||
572 | catch (Exception ex) | ||
573 | { | ||
574 | Console.WriteLine(ex.ToString()); | ||
575 | } | ||
576 | } | ||
577 | |||
578 | public InventoryFolder FindClothingFolder() | ||
579 | { | ||
580 | UUID rootfolder = Client.Inventory.Store.RootFolder.UUID; | ||
581 | List<InventoryBase> listfolders = Client.Inventory.Store.GetContents(rootfolder); | ||
582 | InventoryFolder clothfolder = new InventoryFolder(UUID.Random()); | ||
583 | foreach (InventoryBase folder in listfolders) | ||
584 | { | ||
585 | if (folder.Name == "Clothing") | ||
586 | { | ||
587 | clothfolder = (InventoryFolder)folder; | ||
588 | break; | ||
589 | } | ||
590 | } | ||
591 | return clothfolder; | ||
592 | } | ||
593 | |||
594 | public void Network_LoginProgress(object sender, LoginProgressEventArgs args) | ||
595 | { | ||
596 | m_log.DebugFormat("[BOT]: Bot {0} {1} in Network_LoginProcess", Name, args.Status); | ||
597 | |||
598 | if (args.Status == LoginStatus.Success) | ||
599 | { | ||
600 | if (OnConnected != null) | ||
601 | { | ||
602 | OnConnected(this, EventType.CONNECTED); | ||
603 | } | ||
604 | } | ||
605 | } | ||
606 | |||
607 | public void Network_SimConnected(object sender, SimConnectedEventArgs args) | ||
608 | { | ||
609 | m_log.DebugFormat( | ||
610 | "[BOT]: Bot {0} connected to region {1} at {2}", Name, args.Simulator.Name, args.Simulator.IPEndPoint); | ||
611 | } | ||
612 | |||
613 | public void Network_SimDisconnected(object sender, SimDisconnectedEventArgs args) | ||
614 | { | ||
615 | m_log.DebugFormat( | ||
616 | "[BOT]: Bot {0} disconnected from region {1} at {2}", Name, args.Simulator.Name, args.Simulator.IPEndPoint); | ||
617 | } | ||
618 | |||
619 | public void Network_OnDisconnected(object sender, DisconnectedEventArgs args) | ||
620 | { | ||
621 | ConnectionState = ConnectionState.Disconnected; | ||
622 | |||
623 | m_log.DebugFormat( | ||
624 | "[BOT]: Bot {0} disconnected from grid, reason {1}, message {2}", Name, args.Reason, args.Message); | ||
625 | |||
626 | // m_log.ErrorFormat("Fired Network_OnDisconnected"); | ||
627 | |||
628 | // if ( | ||
629 | // (args.Reason == NetworkManager.DisconnectType.SimShutdown | ||
630 | // || args.Reason == NetworkManager.DisconnectType.NetworkTimeout) | ||
631 | // && OnDisconnected != null) | ||
632 | |||
633 | |||
634 | |||
635 | if ( | ||
636 | (args.Reason == NetworkManager.DisconnectType.ClientInitiated | ||
637 | || args.Reason == NetworkManager.DisconnectType.ServerInitiated | ||
638 | || args.Reason == NetworkManager.DisconnectType.NetworkTimeout) | ||
639 | && OnDisconnected != null) | ||
640 | // if (OnDisconnected != null) | ||
641 | { | ||
642 | OnDisconnected(this, EventType.DISCONNECTED); | ||
643 | } | ||
644 | } | ||
645 | |||
646 | public void Objects_NewPrim(object sender, PrimEventArgs args) | ||
647 | { | ||
648 | if (!RequestObjectTextures) | ||
649 | return; | ||
650 | |||
651 | Primitive prim = args.Prim; | ||
652 | |||
653 | if (prim != null) | ||
654 | { | ||
655 | lock (m_objects) | ||
656 | m_objects[prim.ID] = prim; | ||
657 | |||
658 | if (prim.Textures != null) | ||
659 | { | ||
660 | if (prim.Textures.DefaultTexture.TextureID != UUID.Zero) | ||
661 | { | ||
662 | GetTextureOrMesh(prim.Textures.DefaultTexture.TextureID, true); | ||
663 | } | ||
664 | |||
665 | for (int i = 0; i < prim.Textures.FaceTextures.Length; i++) | ||
666 | { | ||
667 | Primitive.TextureEntryFace face = prim.Textures.FaceTextures[i]; | ||
668 | |||
669 | if (face != null) | ||
670 | { | ||
671 | UUID textureID = prim.Textures.FaceTextures[i].TextureID; | ||
672 | |||
673 | if (textureID != UUID.Zero) | ||
674 | GetTextureOrMesh(textureID, true); | ||
675 | } | ||
676 | } | ||
677 | } | ||
678 | |||
679 | if (prim.Sculpt != null && prim.Sculpt.SculptTexture != UUID.Zero) | ||
680 | { | ||
681 | bool mesh = (prim.Sculpt.Type == SculptType.Mesh); | ||
682 | GetTextureOrMesh(prim.Sculpt.SculptTexture, !mesh); | ||
683 | } | ||
684 | } | ||
685 | } | ||
686 | |||
687 | private void GetTextureOrMesh(UUID assetID, bool texture) | ||
688 | { | ||
689 | lock (Manager.AssetsReceived) | ||
690 | { | ||
691 | // Don't request assets more than once. | ||
692 | if (Manager.AssetsReceived.ContainsKey(assetID)) | ||
693 | return; | ||
694 | |||
695 | Manager.AssetsReceived[assetID] = false; | ||
696 | } | ||
697 | |||
698 | try | ||
699 | { | ||
700 | if (texture) | ||
701 | Client.Assets.RequestImage(assetID, ImageType.Normal, Asset_TextureCallback_Texture); | ||
702 | else | ||
703 | Client.Assets.RequestMesh(assetID, Asset_MeshCallback); | ||
704 | } | ||
705 | catch (Exception e) | ||
706 | { | ||
707 | m_log.Warn(string.Format("Error requesting {0} {1}", texture ? "texture" : "mesh", assetID), e); | ||
708 | } | ||
709 | } | ||
710 | |||
711 | public void Asset_TextureCallback_Texture(TextureRequestState state, AssetTexture assetTexture) | ||
712 | { | ||
713 | if (state == TextureRequestState.Finished) | ||
714 | { | ||
715 | lock (Manager.AssetsReceived) | ||
716 | Manager.AssetsReceived[assetTexture.AssetID] = true; | ||
717 | } | ||
718 | } | ||
719 | |||
720 | private void Asset_MeshCallback(bool success, AssetMesh assetMesh) | ||
721 | { | ||
722 | lock (Manager.AssetsReceived) | ||
723 | Manager.AssetsReceived[assetMesh.AssetID] = success; | ||
724 | } | ||
725 | |||
726 | public void Asset_ReceivedCallback(AssetDownload transfer, Asset asset) | ||
727 | { | ||
728 | lock (Manager.AssetsReceived) | ||
729 | Manager.AssetsReceived[asset.AssetID] = true; | ||
730 | |||
731 | // if (wear == "save") | ||
732 | // { | ||
733 | // SaveAsset((AssetWearable) asset); | ||
734 | // } | ||
735 | } | ||
736 | |||
737 | private void PacketReceivedDebugHandler(object o, PacketReceivedEventArgs args) | ||
738 | { | ||
739 | Packet p = args.Packet; | ||
740 | Header h = p.Header; | ||
741 | Simulator s = args.Simulator; | ||
742 | |||
743 | m_log.DebugFormat( | ||
744 | "[BOT]: Bot {0} received from {1} packet {2} #{3}, rel {4}, res {5}", | ||
745 | Name, s.Name, p.Type, h.Sequence, h.Reliable, h.Resent); | ||
746 | } | ||
747 | } | ||
748 | } | ||
diff --git a/OpenSim/Tools/pCampBot/BotManager.cs b/OpenSim/Tools/pCampBot/BotManager.cs new file mode 100644 index 0000000..0af9592 --- /dev/null +++ b/OpenSim/Tools/pCampBot/BotManager.cs | |||
@@ -0,0 +1,983 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.Collections.Generic; | ||
30 | using System.Linq; | ||
31 | using System.Reflection; | ||
32 | using System.Threading; | ||
33 | using OpenMetaverse; | ||
34 | using log4net; | ||
35 | using log4net.Appender; | ||
36 | using log4net.Core; | ||
37 | using log4net.Repository; | ||
38 | using Nini.Config; | ||
39 | using OpenSim.Framework; | ||
40 | using OpenSim.Framework.Console; | ||
41 | using OpenSim.Framework.Monitoring; | ||
42 | using pCampBot.Interfaces; | ||
43 | |||
44 | namespace pCampBot | ||
45 | { | ||
46 | public enum BotManagerBotConnectingState | ||
47 | { | ||
48 | Initializing, | ||
49 | Ready, | ||
50 | Connecting, | ||
51 | Disconnecting | ||
52 | } | ||
53 | |||
54 | /// <summary> | ||
55 | /// Thread/Bot manager for the application | ||
56 | /// </summary> | ||
57 | public class BotManager | ||
58 | { | ||
59 | private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); | ||
60 | |||
61 | public const int DefaultLoginDelay = 5000; | ||
62 | |||
63 | /// <summary> | ||
64 | /// Is pCampbot ready to connect or currently in the process of connecting or disconnecting bots? | ||
65 | /// </summary> | ||
66 | public BotManagerBotConnectingState BotConnectingState { get; private set; } | ||
67 | |||
68 | /// <summary> | ||
69 | /// Used to control locking as we can't lock an enum. | ||
70 | /// </summary> | ||
71 | private object BotConnectingStateChangeObject = new object(); | ||
72 | |||
73 | /// <summary> | ||
74 | /// Delay between logins of multiple bots. | ||
75 | /// </summary> | ||
76 | /// <remarks>TODO: This value needs to be configurable by a command line argument.</remarks> | ||
77 | public int LoginDelay { get; set; } | ||
78 | |||
79 | /// <summary> | ||
80 | /// Command console | ||
81 | /// </summary> | ||
82 | protected CommandConsole m_console; | ||
83 | |||
84 | /// <summary> | ||
85 | /// Controls whether bots start out sending agent updates on connection. | ||
86 | /// </summary> | ||
87 | public bool InitBotSendAgentUpdates { get; set; } | ||
88 | |||
89 | /// <summary> | ||
90 | /// Controls whether bots request textures for the object information they receive | ||
91 | /// </summary> | ||
92 | public bool InitBotRequestObjectTextures { get; set; } | ||
93 | |||
94 | /// <summary> | ||
95 | /// Created bots, whether active or inactive. | ||
96 | /// </summary> | ||
97 | protected List<Bot> m_bots; | ||
98 | |||
99 | /// <summary> | ||
100 | /// Random number generator. | ||
101 | /// </summary> | ||
102 | public Random Rng { get; private set; } | ||
103 | |||
104 | /// <summary> | ||
105 | /// Track the assets we have and have not received so we don't endlessly repeat requests. | ||
106 | /// </summary> | ||
107 | public Dictionary<UUID, bool> AssetsReceived { get; private set; } | ||
108 | |||
109 | /// <summary> | ||
110 | /// The regions that we know about. | ||
111 | /// </summary> | ||
112 | public Dictionary<ulong, GridRegion> RegionsKnown { get; private set; } | ||
113 | |||
114 | /// <summary> | ||
115 | /// First name for bots | ||
116 | /// </summary> | ||
117 | private string m_firstName; | ||
118 | |||
119 | /// <summary> | ||
120 | /// Last name stem for bots | ||
121 | /// </summary> | ||
122 | private string m_lastNameStem; | ||
123 | |||
124 | /// <summary> | ||
125 | /// Password for bots | ||
126 | /// </summary> | ||
127 | private string m_password; | ||
128 | |||
129 | /// <summary> | ||
130 | /// Login URI for bots. | ||
131 | /// </summary> | ||
132 | private string m_loginUri; | ||
133 | |||
134 | /// <summary> | ||
135 | /// Start location for bots. | ||
136 | /// </summary> | ||
137 | private string m_startUri; | ||
138 | |||
139 | /// <summary> | ||
140 | /// Postfix bot number at which bot sequence starts. | ||
141 | /// </summary> | ||
142 | private int m_fromBotNumber; | ||
143 | |||
144 | /// <summary> | ||
145 | /// Wear setting for bots. | ||
146 | /// </summary> | ||
147 | private string m_wearSetting; | ||
148 | |||
149 | /// <summary> | ||
150 | /// Behaviour switches for bots. | ||
151 | /// </summary> | ||
152 | private HashSet<string> m_defaultBehaviourSwitches = new HashSet<string>(); | ||
153 | |||
154 | /// <summary> | ||
155 | /// Collects general information on this server (which reveals this to be a misnamed class). | ||
156 | /// </summary> | ||
157 | private ServerStatsCollector m_serverStatsCollector; | ||
158 | |||
159 | /// <summary> | ||
160 | /// Constructor Creates MainConsole.Instance to take commands and provide the place to write data | ||
161 | /// </summary> | ||
162 | public BotManager() | ||
163 | { | ||
164 | // We set this to avoid issues with bots running out of HTTP connections if many are run from a single machine | ||
165 | // to multiple regions. | ||
166 | Settings.MAX_HTTP_CONNECTIONS = int.MaxValue; | ||
167 | |||
168 | // System.Threading.ThreadPool.SetMaxThreads(600, 240); | ||
169 | // | ||
170 | // int workerThreads, iocpThreads; | ||
171 | // System.Threading.ThreadPool.GetMaxThreads(out workerThreads, out iocpThreads); | ||
172 | // Console.WriteLine("ThreadPool.GetMaxThreads {0} {1}", workerThreads, iocpThreads); | ||
173 | |||
174 | InitBotSendAgentUpdates = true; | ||
175 | InitBotRequestObjectTextures = true; | ||
176 | |||
177 | LoginDelay = DefaultLoginDelay; | ||
178 | |||
179 | Rng = new Random(Environment.TickCount); | ||
180 | AssetsReceived = new Dictionary<UUID, bool>(); | ||
181 | RegionsKnown = new Dictionary<ulong, GridRegion>(); | ||
182 | |||
183 | m_console = CreateConsole(); | ||
184 | MainConsole.Instance = m_console; | ||
185 | |||
186 | // Make log4net see the console | ||
187 | // | ||
188 | ILoggerRepository repository = LogManager.GetRepository(); | ||
189 | IAppender[] appenders = repository.GetAppenders(); | ||
190 | OpenSimAppender consoleAppender = null; | ||
191 | |||
192 | foreach (IAppender appender in appenders) | ||
193 | { | ||
194 | if (appender.Name == "Console") | ||
195 | { | ||
196 | consoleAppender = (OpenSimAppender)appender; | ||
197 | consoleAppender.Console = m_console; | ||
198 | break; | ||
199 | } | ||
200 | } | ||
201 | |||
202 | m_console.Commands.AddCommand( | ||
203 | "Bots", false, "shutdown", "shutdown", "Shutdown bots and exit", HandleShutdown); | ||
204 | |||
205 | m_console.Commands.AddCommand( | ||
206 | "Bots", false, "quit", "quit", "Shutdown bots and exit", HandleShutdown); | ||
207 | |||
208 | m_console.Commands.AddCommand( | ||
209 | "Bots", false, "connect", "connect [<n>]", "Connect bots", | ||
210 | "If an <n> is given, then the first <n> disconnected bots by postfix number are connected.\n" | ||
211 | + "If no <n> is given, then all currently disconnected bots are connected.", | ||
212 | HandleConnect); | ||
213 | |||
214 | m_console.Commands.AddCommand( | ||
215 | "Bots", false, "disconnect", "disconnect [<n>]", "Disconnect bots", | ||
216 | "Disconnecting bots will interupt any bot connection process, including connection on startup.\n" | ||
217 | + "If an <n> is given, then the last <n> connected bots by postfix number are disconnected.\n" | ||
218 | + "If no <n> is given, then all currently connected bots are disconnected.", | ||
219 | HandleDisconnect); | ||
220 | |||
221 | m_console.Commands.AddCommand( | ||
222 | "Bots", false, "add behaviour", "add behaviour <abbreviated-name> [<bot-number>]", | ||
223 | "Add a behaviour to a bot", | ||
224 | "If no bot number is specified then behaviour is added to all bots.\n" | ||
225 | + "Can be performed on connected or disconnected bots.", | ||
226 | HandleAddBehaviour); | ||
227 | |||
228 | m_console.Commands.AddCommand( | ||
229 | "Bots", false, "remove behaviour", "remove behaviour <abbreviated-name> [<bot-number>]", | ||
230 | "Remove a behaviour from a bot", | ||
231 | "If no bot number is specified then behaviour is added to all bots.\n" | ||
232 | + "Can be performed on connected or disconnected bots.", | ||
233 | HandleRemoveBehaviour); | ||
234 | |||
235 | m_console.Commands.AddCommand( | ||
236 | "Bots", false, "sit", "sit", "Sit all bots on the ground.", | ||
237 | HandleSit); | ||
238 | |||
239 | m_console.Commands.AddCommand( | ||
240 | "Bots", false, "stand", "stand", "Stand all bots.", | ||
241 | HandleStand); | ||
242 | |||
243 | m_console.Commands.AddCommand( | ||
244 | "Bots", false, "set bots", "set bots <key> <value>", "Set a setting for all bots.", HandleSetBots); | ||
245 | |||
246 | m_console.Commands.AddCommand( | ||
247 | "Bots", false, "show regions", "show regions", "Show regions known to bots", HandleShowRegions); | ||
248 | |||
249 | m_console.Commands.AddCommand( | ||
250 | "Bots", false, "show bots", "show bots", "Shows the status of all bots.", HandleShowBotsStatus); | ||
251 | |||
252 | m_console.Commands.AddCommand( | ||
253 | "Bots", false, "show bot", "show bot <bot-number>", | ||
254 | "Shows the detailed status and settings of a particular bot.", HandleShowBotStatus); | ||
255 | |||
256 | m_console.Commands.AddCommand( | ||
257 | "Debug", | ||
258 | false, | ||
259 | "debug lludp packet", | ||
260 | "debug lludp packet <level> <avatar-first-name> <avatar-last-name>", | ||
261 | "Turn on received packet logging.", | ||
262 | "If level > 0 then all received packets that are not duplicates are logged.\n" | ||
263 | + "If level <= 0 then no received packets are logged.", | ||
264 | HandleDebugLludpPacketCommand); | ||
265 | |||
266 | m_console.Commands.AddCommand( | ||
267 | "Bots", false, "show status", "show status", "Shows pCampbot status.", HandleShowStatus); | ||
268 | |||
269 | m_bots = new List<Bot>(); | ||
270 | |||
271 | Watchdog.Enabled = true; | ||
272 | StatsManager.RegisterConsoleCommands(m_console); | ||
273 | |||
274 | m_serverStatsCollector = new ServerStatsCollector(); | ||
275 | m_serverStatsCollector.Initialise(null); | ||
276 | m_serverStatsCollector.Enabled = true; | ||
277 | m_serverStatsCollector.Start(); | ||
278 | |||
279 | BotConnectingState = BotManagerBotConnectingState.Ready; | ||
280 | } | ||
281 | |||
282 | /// <summary> | ||
283 | /// Startup number of bots specified in the starting arguments | ||
284 | /// </summary> | ||
285 | /// <param name="botcount">How many bots to start up</param> | ||
286 | /// <param name="cs">The configuration for the bots to use</param> | ||
287 | public void CreateBots(int botcount, IConfig startupConfig) | ||
288 | { | ||
289 | m_firstName = startupConfig.GetString("firstname"); | ||
290 | m_lastNameStem = startupConfig.GetString("lastname"); | ||
291 | m_password = startupConfig.GetString("password"); | ||
292 | m_loginUri = startupConfig.GetString("loginuri"); | ||
293 | m_fromBotNumber = startupConfig.GetInt("from", 0); | ||
294 | m_wearSetting = startupConfig.GetString("wear", "no"); | ||
295 | |||
296 | m_startUri = ParseInputStartLocationToUri(startupConfig.GetString("start", "last")); | ||
297 | |||
298 | Array.ForEach<string>( | ||
299 | startupConfig.GetString("behaviours", "p").Split(new char[] { ',' }), b => m_defaultBehaviourSwitches.Add(b)); | ||
300 | |||
301 | for (int i = 0; i < botcount; i++) | ||
302 | { | ||
303 | lock (m_bots) | ||
304 | { | ||
305 | string lastName = string.Format("{0}_{1}", m_lastNameStem, i + m_fromBotNumber); | ||
306 | |||
307 | CreateBot( | ||
308 | this, | ||
309 | CreateBehavioursFromAbbreviatedNames(m_defaultBehaviourSwitches), | ||
310 | m_firstName, lastName, m_password, m_loginUri, m_startUri, m_wearSetting); | ||
311 | } | ||
312 | } | ||
313 | } | ||
314 | |||
315 | private List<IBehaviour> CreateBehavioursFromAbbreviatedNames(HashSet<string> abbreviatedNames) | ||
316 | { | ||
317 | // We must give each bot its own list of instantiated behaviours since they store state. | ||
318 | List<IBehaviour> behaviours = new List<IBehaviour>(); | ||
319 | |||
320 | // Hard-coded for now | ||
321 | foreach (string abName in abbreviatedNames) | ||
322 | { | ||
323 | IBehaviour newBehaviour = null; | ||
324 | |||
325 | if (abName == "c") | ||
326 | newBehaviour = new CrossBehaviour(); | ||
327 | |||
328 | if (abName == "g") | ||
329 | newBehaviour = new GrabbingBehaviour(); | ||
330 | |||
331 | if (abName == "n") | ||
332 | newBehaviour = new NoneBehaviour(); | ||
333 | |||
334 | if (abName == "p") | ||
335 | newBehaviour = new PhysicsBehaviour(); | ||
336 | |||
337 | if (abName == "t") | ||
338 | newBehaviour = new TeleportBehaviour(); | ||
339 | |||
340 | if (abName == "tw") | ||
341 | newBehaviour = new TwitchyBehaviour(); | ||
342 | |||
343 | if (abName == "ph2") | ||
344 | newBehaviour = new PhysicsBehaviour2(); | ||
345 | |||
346 | if (abName == "inv") | ||
347 | newBehaviour = new InventoryDownloadBehaviour(); | ||
348 | |||
349 | if (newBehaviour != null) | ||
350 | { | ||
351 | behaviours.Add(newBehaviour); | ||
352 | } | ||
353 | else | ||
354 | { | ||
355 | MainConsole.Instance.OutputFormat("No behaviour with abbreviated name {0} found", abName); | ||
356 | } | ||
357 | } | ||
358 | |||
359 | return behaviours; | ||
360 | } | ||
361 | |||
362 | public void ConnectBots(int botcount) | ||
363 | { | ||
364 | lock (BotConnectingStateChangeObject) | ||
365 | { | ||
366 | if (BotConnectingState != BotManagerBotConnectingState.Ready) | ||
367 | { | ||
368 | MainConsole.Instance.OutputFormat( | ||
369 | "Bot connecting status is {0}. Please wait for previous process to complete.", BotConnectingState); | ||
370 | return; | ||
371 | } | ||
372 | |||
373 | BotConnectingState = BotManagerBotConnectingState.Connecting; | ||
374 | } | ||
375 | |||
376 | Thread connectBotThread = new Thread(o => ConnectBotsInternal(botcount)); | ||
377 | |||
378 | connectBotThread.Name = "Bots connection thread"; | ||
379 | connectBotThread.Start(); | ||
380 | } | ||
381 | |||
382 | private void ConnectBotsInternal(int botCount) | ||
383 | { | ||
384 | m_log.InfoFormat( | ||
385 | "[BOT MANAGER]: Starting {0} bots connecting to {1}, location {2}, named {3} {4}_<n>", | ||
386 | botCount, | ||
387 | m_loginUri, | ||
388 | m_startUri, | ||
389 | m_firstName, | ||
390 | m_lastNameStem); | ||
391 | |||
392 | m_log.DebugFormat("[BOT MANAGER]: Delay between logins is {0}ms", LoginDelay); | ||
393 | m_log.DebugFormat("[BOT MANAGER]: BotsSendAgentUpdates is {0}", InitBotSendAgentUpdates); | ||
394 | m_log.DebugFormat("[BOT MANAGER]: InitBotRequestObjectTextures is {0}", InitBotRequestObjectTextures); | ||
395 | |||
396 | List<Bot> botsToConnect = new List<Bot>(); | ||
397 | |||
398 | lock (m_bots) | ||
399 | { | ||
400 | foreach (Bot bot in m_bots) | ||
401 | { | ||
402 | if (bot.ConnectionState == ConnectionState.Disconnected) | ||
403 | botsToConnect.Add(bot); | ||
404 | |||
405 | if (botsToConnect.Count >= botCount) | ||
406 | break; | ||
407 | } | ||
408 | } | ||
409 | |||
410 | foreach (Bot bot in botsToConnect) | ||
411 | { | ||
412 | lock (BotConnectingStateChangeObject) | ||
413 | { | ||
414 | if (BotConnectingState != BotManagerBotConnectingState.Connecting) | ||
415 | { | ||
416 | MainConsole.Instance.Output( | ||
417 | "[BOT MANAGER]: Aborting bot connection due to user-initiated disconnection"); | ||
418 | return; | ||
419 | } | ||
420 | } | ||
421 | |||
422 | bot.Connect(); | ||
423 | |||
424 | // Stagger logins | ||
425 | Thread.Sleep(LoginDelay); | ||
426 | } | ||
427 | |||
428 | lock (BotConnectingStateChangeObject) | ||
429 | { | ||
430 | if (BotConnectingState == BotManagerBotConnectingState.Connecting) | ||
431 | BotConnectingState = BotManagerBotConnectingState.Ready; | ||
432 | } | ||
433 | } | ||
434 | |||
435 | /// <summary> | ||
436 | /// Parses the command line start location to a start string/uri that the login mechanism will recognize. | ||
437 | /// </summary> | ||
438 | /// <returns> | ||
439 | /// The input start location to URI. | ||
440 | /// </returns> | ||
441 | /// <param name='startLocation'> | ||
442 | /// Start location. | ||
443 | /// </param> | ||
444 | private string ParseInputStartLocationToUri(string startLocation) | ||
445 | { | ||
446 | if (startLocation == "home" || startLocation == "last") | ||
447 | return startLocation; | ||
448 | |||
449 | string regionName; | ||
450 | |||
451 | // Just a region name or only one (!) extra component. Like a viewer, we will stick 128/128/0 on the end | ||
452 | Vector3 startPos = new Vector3(128, 128, 0); | ||
453 | |||
454 | string[] startLocationComponents = startLocation.Split('/'); | ||
455 | |||
456 | regionName = startLocationComponents[0]; | ||
457 | |||
458 | if (startLocationComponents.Length >= 2) | ||
459 | { | ||
460 | float.TryParse(startLocationComponents[1], out startPos.X); | ||
461 | |||
462 | if (startLocationComponents.Length >= 3) | ||
463 | { | ||
464 | float.TryParse(startLocationComponents[2], out startPos.Y); | ||
465 | |||
466 | if (startLocationComponents.Length >= 4) | ||
467 | float.TryParse(startLocationComponents[3], out startPos.Z); | ||
468 | } | ||
469 | } | ||
470 | |||
471 | return string.Format("uri:{0}&{1}&{2}&{3}", regionName, startPos.X, startPos.Y, startPos.Z); | ||
472 | } | ||
473 | |||
474 | /// <summary> | ||
475 | /// This creates a bot but does not start it. | ||
476 | /// </summary> | ||
477 | /// <param name="bm"></param> | ||
478 | /// <param name="behaviours">Behaviours for this bot to perform.</param> | ||
479 | /// <param name="firstName">First name</param> | ||
480 | /// <param name="lastName">Last name</param> | ||
481 | /// <param name="password">Password</param> | ||
482 | /// <param name="loginUri">Login URI</param> | ||
483 | /// <param name="startLocation">Location to start the bot. Can be "last", "home" or a specific sim name.</param> | ||
484 | /// <param name="wearSetting"></param> | ||
485 | public void CreateBot( | ||
486 | BotManager bm, List<IBehaviour> behaviours, | ||
487 | string firstName, string lastName, string password, string loginUri, string startLocation, string wearSetting) | ||
488 | { | ||
489 | MainConsole.Instance.OutputFormat( | ||
490 | "[BOT MANAGER]: Creating bot {0} {1}, behaviours are {2}", | ||
491 | firstName, lastName, string.Join(",", behaviours.ConvertAll<string>(b => b.Name).ToArray())); | ||
492 | |||
493 | Bot pb = new Bot(bm, behaviours, firstName, lastName, password, startLocation, loginUri); | ||
494 | pb.wear = wearSetting; | ||
495 | pb.Client.Settings.SEND_AGENT_UPDATES = InitBotSendAgentUpdates; | ||
496 | pb.RequestObjectTextures = InitBotRequestObjectTextures; | ||
497 | |||
498 | pb.OnConnected += handlebotEvent; | ||
499 | pb.OnDisconnected += handlebotEvent; | ||
500 | |||
501 | m_bots.Add(pb); | ||
502 | } | ||
503 | |||
504 | /// <summary> | ||
505 | /// High level connnected/disconnected events so we can keep track of our threads by proxy | ||
506 | /// </summary> | ||
507 | /// <param name="callbot"></param> | ||
508 | /// <param name="eventt"></param> | ||
509 | private void handlebotEvent(Bot callbot, EventType eventt) | ||
510 | { | ||
511 | switch (eventt) | ||
512 | { | ||
513 | case EventType.CONNECTED: | ||
514 | { | ||
515 | m_log.Info("[" + callbot.FirstName + " " + callbot.LastName + "]: Connected"); | ||
516 | break; | ||
517 | } | ||
518 | |||
519 | case EventType.DISCONNECTED: | ||
520 | { | ||
521 | m_log.Info("[" + callbot.FirstName + " " + callbot.LastName + "]: Disconnected"); | ||
522 | break; | ||
523 | } | ||
524 | } | ||
525 | } | ||
526 | |||
527 | /// <summary> | ||
528 | /// Standard CreateConsole routine | ||
529 | /// </summary> | ||
530 | /// <returns></returns> | ||
531 | protected CommandConsole CreateConsole() | ||
532 | { | ||
533 | return new LocalConsole("pCampbot"); | ||
534 | } | ||
535 | |||
536 | private void HandleConnect(string module, string[] cmd) | ||
537 | { | ||
538 | lock (m_bots) | ||
539 | { | ||
540 | int botsToConnect; | ||
541 | int disconnectedBots = m_bots.Count(b => b.ConnectionState == ConnectionState.Disconnected); | ||
542 | |||
543 | if (cmd.Length == 1) | ||
544 | { | ||
545 | botsToConnect = disconnectedBots; | ||
546 | } | ||
547 | else | ||
548 | { | ||
549 | if (!ConsoleUtil.TryParseConsoleNaturalInt(MainConsole.Instance, cmd[1], out botsToConnect)) | ||
550 | return; | ||
551 | |||
552 | botsToConnect = Math.Min(botsToConnect, disconnectedBots); | ||
553 | } | ||
554 | |||
555 | MainConsole.Instance.OutputFormat("Connecting {0} bots", botsToConnect); | ||
556 | |||
557 | ConnectBots(botsToConnect); | ||
558 | } | ||
559 | } | ||
560 | |||
561 | private void HandleAddBehaviour(string module, string[] cmd) | ||
562 | { | ||
563 | if (cmd.Length < 3 || cmd.Length > 4) | ||
564 | { | ||
565 | MainConsole.Instance.OutputFormat("Usage: add behaviour <abbreviated-behaviour> [<bot-number>]"); | ||
566 | return; | ||
567 | } | ||
568 | |||
569 | string rawBehaviours = cmd[2]; | ||
570 | |||
571 | List<Bot> botsToEffect = new List<Bot>(); | ||
572 | |||
573 | if (cmd.Length == 3) | ||
574 | { | ||
575 | lock (m_bots) | ||
576 | botsToEffect.AddRange(m_bots); | ||
577 | } | ||
578 | else | ||
579 | { | ||
580 | int botNumber; | ||
581 | if (!ConsoleUtil.TryParseConsoleNaturalInt(MainConsole.Instance, cmd[3], out botNumber)) | ||
582 | return; | ||
583 | |||
584 | Bot bot = GetBotFromNumber(botNumber); | ||
585 | |||
586 | if (bot == null) | ||
587 | { | ||
588 | MainConsole.Instance.OutputFormat("Error: No bot found with number {0}", botNumber); | ||
589 | return; | ||
590 | } | ||
591 | |||
592 | botsToEffect.Add(bot); | ||
593 | } | ||
594 | |||
595 | |||
596 | HashSet<string> rawAbbreviatedSwitchesToAdd = new HashSet<string>(); | ||
597 | Array.ForEach<string>(rawBehaviours.Split(new char[] { ',' }), b => rawAbbreviatedSwitchesToAdd.Add(b)); | ||
598 | |||
599 | foreach (Bot bot in botsToEffect) | ||
600 | { | ||
601 | List<IBehaviour> behavioursAdded = new List<IBehaviour>(); | ||
602 | |||
603 | foreach (IBehaviour behaviour in CreateBehavioursFromAbbreviatedNames(rawAbbreviatedSwitchesToAdd)) | ||
604 | { | ||
605 | if (bot.AddBehaviour(behaviour)) | ||
606 | behavioursAdded.Add(behaviour); | ||
607 | } | ||
608 | |||
609 | MainConsole.Instance.OutputFormat( | ||
610 | "Added behaviours {0} to bot {1}", | ||
611 | string.Join(", ", behavioursAdded.ConvertAll<string>(b => b.Name).ToArray()), bot.Name); | ||
612 | } | ||
613 | } | ||
614 | |||
615 | private void HandleRemoveBehaviour(string module, string[] cmd) | ||
616 | { | ||
617 | if (cmd.Length < 3 || cmd.Length > 4) | ||
618 | { | ||
619 | MainConsole.Instance.OutputFormat("Usage: remove behaviour <abbreviated-behaviour> [<bot-number>]"); | ||
620 | return; | ||
621 | } | ||
622 | |||
623 | string rawBehaviours = cmd[2]; | ||
624 | |||
625 | List<Bot> botsToEffect = new List<Bot>(); | ||
626 | |||
627 | if (cmd.Length == 3) | ||
628 | { | ||
629 | lock (m_bots) | ||
630 | botsToEffect.AddRange(m_bots); | ||
631 | } | ||
632 | else | ||
633 | { | ||
634 | int botNumber; | ||
635 | if (!ConsoleUtil.TryParseConsoleNaturalInt(MainConsole.Instance, cmd[3], out botNumber)) | ||
636 | return; | ||
637 | |||
638 | Bot bot = GetBotFromNumber(botNumber); | ||
639 | |||
640 | if (bot == null) | ||
641 | { | ||
642 | MainConsole.Instance.OutputFormat("Error: No bot found with number {0}", botNumber); | ||
643 | return; | ||
644 | } | ||
645 | |||
646 | botsToEffect.Add(bot); | ||
647 | } | ||
648 | |||
649 | HashSet<string> abbreviatedBehavioursToRemove = new HashSet<string>(); | ||
650 | Array.ForEach<string>(rawBehaviours.Split(new char[] { ',' }), b => abbreviatedBehavioursToRemove.Add(b)); | ||
651 | |||
652 | foreach (Bot bot in botsToEffect) | ||
653 | { | ||
654 | List<IBehaviour> behavioursRemoved = new List<IBehaviour>(); | ||
655 | |||
656 | foreach (string b in abbreviatedBehavioursToRemove) | ||
657 | { | ||
658 | IBehaviour behaviour; | ||
659 | |||
660 | if (bot.TryGetBehaviour(b, out behaviour)) | ||
661 | { | ||
662 | bot.RemoveBehaviour(b); | ||
663 | behavioursRemoved.Add(behaviour); | ||
664 | } | ||
665 | } | ||
666 | |||
667 | MainConsole.Instance.OutputFormat( | ||
668 | "Removed behaviours {0} from bot {1}", | ||
669 | string.Join(", ", behavioursRemoved.ConvertAll<string>(b => b.Name).ToArray()), bot.Name); | ||
670 | } | ||
671 | } | ||
672 | |||
673 | private void HandleDisconnect(string module, string[] cmd) | ||
674 | { | ||
675 | List<Bot> connectedBots; | ||
676 | int botsToDisconnectCount; | ||
677 | |||
678 | lock (m_bots) | ||
679 | connectedBots = m_bots.FindAll(b => b.ConnectionState == ConnectionState.Connected); | ||
680 | |||
681 | if (cmd.Length == 1) | ||
682 | { | ||
683 | botsToDisconnectCount = connectedBots.Count; | ||
684 | } | ||
685 | else | ||
686 | { | ||
687 | if (!ConsoleUtil.TryParseConsoleNaturalInt(MainConsole.Instance, cmd[1], out botsToDisconnectCount)) | ||
688 | return; | ||
689 | |||
690 | botsToDisconnectCount = Math.Min(botsToDisconnectCount, connectedBots.Count); | ||
691 | } | ||
692 | |||
693 | lock (BotConnectingStateChangeObject) | ||
694 | BotConnectingState = BotManagerBotConnectingState.Disconnecting; | ||
695 | |||
696 | Thread disconnectBotThread = new Thread(o => DisconnectBotsInternal(connectedBots, botsToDisconnectCount)); | ||
697 | |||
698 | disconnectBotThread.Name = "Bots disconnection thread"; | ||
699 | disconnectBotThread.Start(); | ||
700 | } | ||
701 | |||
702 | private void DisconnectBotsInternal(List<Bot> connectedBots, int disconnectCount) | ||
703 | { | ||
704 | MainConsole.Instance.OutputFormat("Disconnecting {0} bots", disconnectCount); | ||
705 | |||
706 | int disconnectedBots = 0; | ||
707 | |||
708 | for (int i = connectedBots.Count - 1; i >= 0; i--) | ||
709 | { | ||
710 | if (disconnectedBots >= disconnectCount) | ||
711 | break; | ||
712 | |||
713 | Bot thisBot = connectedBots[i]; | ||
714 | |||
715 | if (thisBot.ConnectionState == ConnectionState.Connected) | ||
716 | { | ||
717 | ThreadPool.QueueUserWorkItem(o => thisBot.Disconnect()); | ||
718 | disconnectedBots++; | ||
719 | } | ||
720 | } | ||
721 | |||
722 | lock (BotConnectingStateChangeObject) | ||
723 | BotConnectingState = BotManagerBotConnectingState.Ready; | ||
724 | } | ||
725 | |||
726 | private void HandleSit(string module, string[] cmd) | ||
727 | { | ||
728 | lock (m_bots) | ||
729 | { | ||
730 | foreach (Bot bot in m_bots) | ||
731 | { | ||
732 | if (bot.ConnectionState == ConnectionState.Connected) | ||
733 | { | ||
734 | MainConsole.Instance.OutputFormat("Sitting bot {0} on ground.", bot.Name); | ||
735 | bot.SitOnGround(); | ||
736 | } | ||
737 | } | ||
738 | } | ||
739 | } | ||
740 | |||
741 | private void HandleStand(string module, string[] cmd) | ||
742 | { | ||
743 | lock (m_bots) | ||
744 | { | ||
745 | foreach (Bot bot in m_bots) | ||
746 | { | ||
747 | if (bot.ConnectionState == ConnectionState.Connected) | ||
748 | { | ||
749 | MainConsole.Instance.OutputFormat("Standing bot {0} from ground.", bot.Name); | ||
750 | bot.Stand(); | ||
751 | } | ||
752 | } | ||
753 | } | ||
754 | } | ||
755 | |||
756 | private void HandleShutdown(string module, string[] cmd) | ||
757 | { | ||
758 | lock (m_bots) | ||
759 | { | ||
760 | int connectedBots = m_bots.Count(b => b.ConnectionState == ConnectionState.Connected); | ||
761 | |||
762 | if (connectedBots > 0) | ||
763 | { | ||
764 | MainConsole.Instance.OutputFormat("Please disconnect {0} connected bots first", connectedBots); | ||
765 | return; | ||
766 | } | ||
767 | } | ||
768 | |||
769 | MainConsole.Instance.Output("Shutting down"); | ||
770 | |||
771 | m_serverStatsCollector.Close(); | ||
772 | |||
773 | Environment.Exit(0); | ||
774 | } | ||
775 | |||
776 | private void HandleSetBots(string module, string[] cmd) | ||
777 | { | ||
778 | string key = cmd[2]; | ||
779 | string rawValue = cmd[3]; | ||
780 | |||
781 | if (key == "SEND_AGENT_UPDATES") | ||
782 | { | ||
783 | bool newSendAgentUpdatesSetting; | ||
784 | |||
785 | if (!ConsoleUtil.TryParseConsoleBool(MainConsole.Instance, rawValue, out newSendAgentUpdatesSetting)) | ||
786 | return; | ||
787 | |||
788 | MainConsole.Instance.OutputFormat( | ||
789 | "Setting SEND_AGENT_UPDATES to {0} for all bots", newSendAgentUpdatesSetting); | ||
790 | |||
791 | lock (m_bots) | ||
792 | m_bots.ForEach(b => b.Client.Settings.SEND_AGENT_UPDATES = newSendAgentUpdatesSetting); | ||
793 | } | ||
794 | else | ||
795 | { | ||
796 | MainConsole.Instance.Output("Error: Only setting currently available is SEND_AGENT_UPDATES"); | ||
797 | } | ||
798 | } | ||
799 | |||
800 | private void HandleDebugLludpPacketCommand(string module, string[] args) | ||
801 | { | ||
802 | if (args.Length != 6) | ||
803 | { | ||
804 | MainConsole.Instance.OutputFormat("Usage: debug lludp packet <level> <bot-first-name> <bot-last-name>"); | ||
805 | return; | ||
806 | } | ||
807 | |||
808 | int level; | ||
809 | |||
810 | if (!ConsoleUtil.TryParseConsoleInt(MainConsole.Instance, args[3], out level)) | ||
811 | return; | ||
812 | |||
813 | string botFirstName = args[4]; | ||
814 | string botLastName = args[5]; | ||
815 | |||
816 | Bot bot; | ||
817 | |||
818 | lock (m_bots) | ||
819 | bot = m_bots.FirstOrDefault(b => b.FirstName == botFirstName && b.LastName == botLastName); | ||
820 | |||
821 | if (bot == null) | ||
822 | { | ||
823 | MainConsole.Instance.OutputFormat("No bot named {0} {1}", botFirstName, botLastName); | ||
824 | return; | ||
825 | } | ||
826 | |||
827 | bot.PacketDebugLevel = level; | ||
828 | |||
829 | MainConsole.Instance.OutputFormat("Set debug level of {0} to {1}", bot.Name, bot.PacketDebugLevel); | ||
830 | } | ||
831 | |||
832 | private void HandleShowRegions(string module, string[] cmd) | ||
833 | { | ||
834 | string outputFormat = "{0,-30} {1, -20} {2, -5} {3, -5}"; | ||
835 | MainConsole.Instance.OutputFormat(outputFormat, "Name", "Handle", "X", "Y"); | ||
836 | |||
837 | lock (RegionsKnown) | ||
838 | { | ||
839 | foreach (GridRegion region in RegionsKnown.Values) | ||
840 | { | ||
841 | MainConsole.Instance.OutputFormat( | ||
842 | outputFormat, region.Name, region.RegionHandle, region.X, region.Y); | ||
843 | } | ||
844 | } | ||
845 | } | ||
846 | |||
847 | private void HandleShowStatus(string module, string[] cmd) | ||
848 | { | ||
849 | ConsoleDisplayList cdl = new ConsoleDisplayList(); | ||
850 | cdl.AddRow("Bot connecting state", BotConnectingState); | ||
851 | |||
852 | MainConsole.Instance.Output(cdl.ToString()); | ||
853 | } | ||
854 | |||
855 | private void HandleShowBotsStatus(string module, string[] cmd) | ||
856 | { | ||
857 | ConsoleDisplayTable cdt = new ConsoleDisplayTable(); | ||
858 | cdt.AddColumn("Name", 24); | ||
859 | cdt.AddColumn("Region", 24); | ||
860 | cdt.AddColumn("Status", 13); | ||
861 | cdt.AddColumn("Conns", 5); | ||
862 | cdt.AddColumn("Behaviours", 20); | ||
863 | |||
864 | Dictionary<ConnectionState, int> totals = new Dictionary<ConnectionState, int>(); | ||
865 | foreach (object o in Enum.GetValues(typeof(ConnectionState))) | ||
866 | totals[(ConnectionState)o] = 0; | ||
867 | |||
868 | lock (m_bots) | ||
869 | { | ||
870 | foreach (Bot bot in m_bots) | ||
871 | { | ||
872 | Simulator currentSim = bot.Client.Network.CurrentSim; | ||
873 | totals[bot.ConnectionState]++; | ||
874 | |||
875 | cdt.AddRow( | ||
876 | bot.Name, | ||
877 | currentSim != null ? currentSim.Name : "(none)", | ||
878 | bot.ConnectionState, | ||
879 | bot.SimulatorsCount, | ||
880 | string.Join(",", bot.Behaviours.Keys.ToArray())); | ||
881 | } | ||
882 | } | ||
883 | |||
884 | MainConsole.Instance.Output(cdt.ToString()); | ||
885 | |||
886 | ConsoleDisplayList cdl = new ConsoleDisplayList(); | ||
887 | |||
888 | foreach (KeyValuePair<ConnectionState, int> kvp in totals) | ||
889 | cdl.AddRow(kvp.Key, kvp.Value); | ||
890 | |||
891 | MainConsole.Instance.Output(cdl.ToString()); | ||
892 | } | ||
893 | |||
894 | private void HandleShowBotStatus(string module, string[] cmd) | ||
895 | { | ||
896 | if (cmd.Length != 3) | ||
897 | { | ||
898 | MainConsole.Instance.Output("Usage: show bot <n>"); | ||
899 | return; | ||
900 | } | ||
901 | |||
902 | int botNumber; | ||
903 | |||
904 | if (!ConsoleUtil.TryParseConsoleInt(MainConsole.Instance, cmd[2], out botNumber)) | ||
905 | return; | ||
906 | |||
907 | Bot bot = GetBotFromNumber(botNumber); | ||
908 | |||
909 | if (bot == null) | ||
910 | { | ||
911 | MainConsole.Instance.OutputFormat("Error: No bot found with number {0}", botNumber); | ||
912 | return; | ||
913 | } | ||
914 | |||
915 | ConsoleDisplayList cdl = new ConsoleDisplayList(); | ||
916 | cdl.AddRow("Name", bot.Name); | ||
917 | cdl.AddRow("Status", bot.ConnectionState); | ||
918 | |||
919 | Simulator currentSim = bot.Client.Network.CurrentSim; | ||
920 | cdl.AddRow("Region", currentSim != null ? currentSim.Name : "(none)"); | ||
921 | |||
922 | List<Simulator> connectedSimulators = bot.Simulators; | ||
923 | List<string> simulatorNames = connectedSimulators.ConvertAll<string>(cs => cs.Name); | ||
924 | cdl.AddRow("Connections", string.Join(", ", simulatorNames.ToArray())); | ||
925 | |||
926 | MainConsole.Instance.Output(cdl.ToString()); | ||
927 | |||
928 | MainConsole.Instance.Output("Settings"); | ||
929 | |||
930 | ConsoleDisplayList statusCdl = new ConsoleDisplayList(); | ||
931 | |||
932 | statusCdl.AddRow( | ||
933 | "Behaviours", | ||
934 | string.Join(", ", bot.Behaviours.Values.ToList().ConvertAll<string>(b => b.Name).ToArray())); | ||
935 | |||
936 | GridClient botClient = bot.Client; | ||
937 | statusCdl.AddRow("SEND_AGENT_UPDATES", botClient.Settings.SEND_AGENT_UPDATES); | ||
938 | |||
939 | MainConsole.Instance.Output(statusCdl.ToString()); | ||
940 | } | ||
941 | |||
942 | /// <summary> | ||
943 | /// Get a specific bot from its number. | ||
944 | /// </summary> | ||
945 | /// <returns>null if no bot was found</returns> | ||
946 | /// <param name='botNumber'></param> | ||
947 | private Bot GetBotFromNumber(int botNumber) | ||
948 | { | ||
949 | string name = GenerateBotNameFromNumber(botNumber); | ||
950 | |||
951 | Bot bot; | ||
952 | |||
953 | lock (m_bots) | ||
954 | bot = m_bots.Find(b => b.Name == name); | ||
955 | |||
956 | return bot; | ||
957 | } | ||
958 | |||
959 | private string GenerateBotNameFromNumber(int botNumber) | ||
960 | { | ||
961 | return string.Format("{0} {1}_{2}", m_firstName, m_lastNameStem, botNumber); | ||
962 | } | ||
963 | |||
964 | internal void Grid_GridRegion(object o, GridRegionEventArgs args) | ||
965 | { | ||
966 | lock (RegionsKnown) | ||
967 | { | ||
968 | GridRegion newRegion = args.Region; | ||
969 | |||
970 | if (RegionsKnown.ContainsKey(newRegion.RegionHandle)) | ||
971 | { | ||
972 | return; | ||
973 | } | ||
974 | else | ||
975 | { | ||
976 | m_log.DebugFormat( | ||
977 | "[BOT MANAGER]: Adding {0} {1} to known regions", newRegion.Name, newRegion.RegionHandle); | ||
978 | RegionsKnown[newRegion.RegionHandle] = newRegion; | ||
979 | } | ||
980 | } | ||
981 | } | ||
982 | } | ||
983 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/pCampBot/Interfaces/IBehaviour.cs b/OpenSim/Tools/pCampBot/Interfaces/IBehaviour.cs new file mode 100644 index 0000000..660c630 --- /dev/null +++ b/OpenSim/Tools/pCampBot/Interfaces/IBehaviour.cs | |||
@@ -0,0 +1,75 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | |||
30 | namespace pCampBot.Interfaces | ||
31 | { | ||
32 | public interface IBehaviour | ||
33 | { | ||
34 | /// <summary> | ||
35 | /// Abbreviated name of this behaviour. | ||
36 | /// </summary> | ||
37 | string AbbreviatedName { get; } | ||
38 | |||
39 | /// <summary> | ||
40 | /// Name of this behaviour. | ||
41 | /// </summary> | ||
42 | string Name { get; } | ||
43 | |||
44 | /// <summary> | ||
45 | /// Initialize the behaviour for this bot. | ||
46 | /// </summary> | ||
47 | /// <remarks> | ||
48 | /// This must be invoked before Action() is called. | ||
49 | /// </remarks> | ||
50 | /// <param name="bot"></param> | ||
51 | void Initialize(Bot bot); | ||
52 | |||
53 | /// <summary> | ||
54 | /// Interrupt the behaviour. | ||
55 | /// </summary> | ||
56 | /// <remarks> | ||
57 | /// This should cause the current Action call() to terminate if this is active. | ||
58 | /// </remarks> | ||
59 | void Interrupt(); | ||
60 | |||
61 | /// <summary> | ||
62 | /// Close down this behaviour. | ||
63 | /// </summary> | ||
64 | /// <remarks> | ||
65 | /// This is triggered if a behaviour is removed via explicit command and when a bot is disconnected | ||
66 | /// </remarks> | ||
67 | void Close(); | ||
68 | |||
69 | /// <summary> | ||
70 | /// Action to take when this behaviour is invoked. | ||
71 | /// </summary> | ||
72 | /// <param name="bot"></param> | ||
73 | void Action(); | ||
74 | } | ||
75 | } \ No newline at end of file | ||
diff --git a/OpenSim/Tools/pCampBot/Properties/AssemblyInfo.cs b/OpenSim/Tools/pCampBot/Properties/AssemblyInfo.cs new file mode 100644 index 0000000..87af19a --- /dev/null +++ b/OpenSim/Tools/pCampBot/Properties/AssemblyInfo.cs | |||
@@ -0,0 +1,33 @@ | |||
1 | using System.Reflection; | ||
2 | using System.Runtime.CompilerServices; | ||
3 | using System.Runtime.InteropServices; | ||
4 | |||
5 | // General Information about an assembly is controlled through the following | ||
6 | // set of attributes. Change these attribute values to modify the information | ||
7 | // associated with an assembly. | ||
8 | [assembly: AssemblyTitle("pCampBot")] | ||
9 | [assembly: AssemblyDescription("")] | ||
10 | [assembly: AssemblyConfiguration("")] | ||
11 | [assembly: AssemblyCompany("http://opensimulator.org")] | ||
12 | [assembly: AssemblyProduct("OpenSim")] | ||
13 | [assembly: AssemblyCopyright("OpenSimulator developers")] | ||
14 | [assembly: AssemblyTrademark("")] | ||
15 | [assembly: AssemblyCulture("")] | ||
16 | |||
17 | // Setting ComVisible to false makes the types in this assembly not visible | ||
18 | // to COM components. If you need to access a type in this assembly from | ||
19 | // COM, set the ComVisible attribute to true on that type. | ||
20 | [assembly: ComVisible(false)] | ||
21 | |||
22 | // The following GUID is for the ID of the typelib if this project is exposed to COM | ||
23 | [assembly: Guid("84a69c60-76d3-4846-bd5b-0e1083774039")] | ||
24 | |||
25 | // Version information for an assembly consists of the following four values: | ||
26 | // | ||
27 | // Major Version | ||
28 | // Minor Version | ||
29 | // Build Number | ||
30 | // Revision | ||
31 | // | ||
32 | [assembly: AssemblyVersion("0.8.2.*")] | ||
33 | |||
diff --git a/OpenSim/Tools/pCampBot/README.txt b/OpenSim/Tools/pCampBot/README.txt new file mode 100644 index 0000000..c4fcf33 --- /dev/null +++ b/OpenSim/Tools/pCampBot/README.txt | |||
@@ -0,0 +1,35 @@ | |||
1 | This is the PhysicsCamperbot libslBot tester. | ||
2 | |||
3 | This is designed to stress test the simulator. It creates <N> | ||
4 | clients that log in, randomly jump/walk around, and can say excuses from | ||
5 | the BOFH. | ||
6 | |||
7 | Bots must have accounts already created. Each bot will have the same firstname and password | ||
8 | but their lastname will be appended with _<bot-number> starting from 0. So if you have two bots called ima bot, their | ||
9 | first names will be ima_bot_0 and ima_bot_1. | ||
10 | |||
11 | *** WARNING *** | ||
12 | Using this bot on a public grid could get you banned permanently, so | ||
13 | just say No! to griefing! | ||
14 | |||
15 | ----- Setup ----- | ||
16 | Linux: To build, in the main opensim directory, run: | ||
17 | ./runprebuild.sh | ||
18 | nant | ||
19 | |||
20 | Windows: Run the prebuild.bat in the main opensim directory and then | ||
21 | open the created solution and compile it. | ||
22 | |||
23 | pCampBot.exe will end up in the regular opensim/bin folder | ||
24 | |||
25 | ----- Running the bot ----- | ||
26 | |||
27 | windows: pCampBot.exe -botcount <N> -loginuri <URI> -firstname <bot-first-name> -lastname <bot-last-name-stem> -password <bot-password> | ||
28 | *nix: mono pCampBot.exe -botcount <N> -loginuri <URI> -firstname <bot-first-name> -lastname <bot-last-name-stem> -password <bot-password> | ||
29 | |||
30 | ----- Commands ----- | ||
31 | |||
32 | The bot has console commands: | ||
33 | help - lists the console commands and what they do | ||
34 | shutdown - gracefully shuts down the bots | ||
35 | quit - forcefully shuts things down leaving stuff unclean | ||
diff --git a/OpenSim/Tools/pCampBot/pCampBot.cs b/OpenSim/Tools/pCampBot/pCampBot.cs new file mode 100644 index 0000000..1fb0e03 --- /dev/null +++ b/OpenSim/Tools/pCampBot/pCampBot.cs | |||
@@ -0,0 +1,175 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.IO; | ||
30 | using System.Reflection; | ||
31 | using System.Threading; | ||
32 | using log4net; | ||
33 | using log4net.Config; | ||
34 | using Nini.Config; | ||
35 | using OpenSim.Framework; | ||
36 | using OpenSim.Framework.Console; | ||
37 | |||
38 | namespace pCampBot | ||
39 | { | ||
40 | /// <summary> | ||
41 | /// Event Types from the BOT. Add new events here | ||
42 | /// </summary> | ||
43 | public enum EventType:int | ||
44 | { | ||
45 | NONE = 0, | ||
46 | CONNECTED = 1, | ||
47 | DISCONNECTED = 2 | ||
48 | } | ||
49 | |||
50 | public class pCampBot | ||
51 | { | ||
52 | private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); | ||
53 | |||
54 | public const string ConfigFileName = "pCampBot.ini"; | ||
55 | |||
56 | [STAThread] | ||
57 | public static void Main(string[] args) | ||
58 | { | ||
59 | XmlConfigurator.Configure(); | ||
60 | |||
61 | IConfig commandLineConfig = ParseConfig(args); | ||
62 | if (commandLineConfig.Get("help") != null || commandLineConfig.Get("loginuri") == null) | ||
63 | { | ||
64 | Help(); | ||
65 | } | ||
66 | else if ( | ||
67 | commandLineConfig.Get("firstname") == null | ||
68 | || commandLineConfig.Get("lastname") == null | ||
69 | || commandLineConfig.Get("password") == null) | ||
70 | { | ||
71 | Console.WriteLine("ERROR: You must supply a firstname, lastname and password for the bots."); | ||
72 | } | ||
73 | else | ||
74 | { | ||
75 | BotManager bm = new BotManager(); | ||
76 | |||
77 | string iniFilePath = Path.GetFullPath(Path.Combine(Util.configDir(), ConfigFileName)); | ||
78 | |||
79 | if (File.Exists(iniFilePath)) | ||
80 | { | ||
81 | m_log.InfoFormat("[PCAMPBOT]: Reading configuration settings from {0}", iniFilePath); | ||
82 | |||
83 | IConfigSource configSource = new IniConfigSource(iniFilePath); | ||
84 | |||
85 | IConfig botManagerConfig = configSource.Configs["BotManager"]; | ||
86 | |||
87 | if (botManagerConfig != null) | ||
88 | { | ||
89 | bm.LoginDelay = botManagerConfig.GetInt("LoginDelay", bm.LoginDelay); | ||
90 | } | ||
91 | |||
92 | IConfig botConfig = configSource.Configs["Bot"]; | ||
93 | |||
94 | if (botConfig != null) | ||
95 | { | ||
96 | bm.InitBotSendAgentUpdates | ||
97 | = botConfig.GetBoolean("SendAgentUpdates", bm.InitBotSendAgentUpdates); | ||
98 | bm.InitBotRequestObjectTextures | ||
99 | = botConfig.GetBoolean("RequestObjectTextures", bm.InitBotRequestObjectTextures); | ||
100 | } | ||
101 | } | ||
102 | |||
103 | int botcount = commandLineConfig.GetInt("botcount", 1); | ||
104 | bool startConnected = commandLineConfig.Get("connect") != null; | ||
105 | |||
106 | bm.CreateBots(botcount, commandLineConfig); | ||
107 | |||
108 | if (startConnected) | ||
109 | bm.ConnectBots(botcount); | ||
110 | |||
111 | while (true) | ||
112 | { | ||
113 | try | ||
114 | { | ||
115 | MainConsole.Instance.Prompt(); | ||
116 | } | ||
117 | catch (Exception e) | ||
118 | { | ||
119 | m_log.ErrorFormat("Command error: {0}", e); | ||
120 | } | ||
121 | } | ||
122 | } | ||
123 | } | ||
124 | |||
125 | private static IConfig ParseConfig(String[] args) | ||
126 | { | ||
127 | //Set up our nifty config.. thanks to nini | ||
128 | ArgvConfigSource cs = new ArgvConfigSource(args); | ||
129 | |||
130 | cs.AddSwitch("Startup", "connect", "c"); | ||
131 | cs.AddSwitch("Startup", "botcount", "n"); | ||
132 | cs.AddSwitch("Startup", "from", "f"); | ||
133 | cs.AddSwitch("Startup", "loginuri", "l"); | ||
134 | cs.AddSwitch("Startup", "start", "s"); | ||
135 | cs.AddSwitch("Startup", "firstname"); | ||
136 | cs.AddSwitch("Startup", "lastname"); | ||
137 | cs.AddSwitch("Startup", "password"); | ||
138 | cs.AddSwitch("Startup", "behaviours", "b"); | ||
139 | cs.AddSwitch("Startup", "help", "h"); | ||
140 | cs.AddSwitch("Startup", "wear"); | ||
141 | |||
142 | IConfig ol = cs.Configs["Startup"]; | ||
143 | return ol; | ||
144 | } | ||
145 | |||
146 | private static void Help() | ||
147 | { | ||
148 | // Added the wear command. This allows the bot to wear real clothes instead of default locked ones. | ||
149 | // You can either say no, to not load anything, yes, to load one of the default wearables, a folder | ||
150 | // name, to load an specific folder, or save, to save an avatar with some already existing wearables | ||
151 | // worn to the folder MyAppearance/FirstName_LastName, and the load it. | ||
152 | |||
153 | Console.WriteLine( | ||
154 | "Usage: pCampBot -loginuri <loginuri> -firstname <first-name> -lastname <last-name> -password <password> [OPTIONS]\n" | ||
155 | + "Spawns a set of bots to test an OpenSim region\n\n" | ||
156 | + " -l, -loginuri loginuri for grid/standalone (required)\n" | ||
157 | + " -s, -start start location for bots (default: last) (optional). Can be \"last\", \"home\" or a specific location with or without co-ords (e.g. \"region1\" or \"region2/50/30/90\"\n" | ||
158 | + " -firstname first name for the bots (required)\n" | ||
159 | + " -lastname lastname for the bots (required). Each lastname will have _<bot-number> appended, e.g. Ima Bot_0\n" | ||
160 | + " -password password for the bots (required)\n" | ||
161 | + " -n, -botcount number of bots to start (default: 1) (optional)\n" | ||
162 | + " -f, -from starting number for login bot names, e.g. 25 will login Ima Bot_25, Ima Bot_26, etc. (default: 0) (optional)\n" | ||
163 | + " -c, -connect connect all bots at startup (optional)\n" | ||
164 | + " -b, behaviours behaviours for bots. Comma separated, e.g. p,g (default: p) (optional)\n" | ||
165 | + " current options are:\n" | ||
166 | + " p (physics - bots constantly move and jump around)\n" | ||
167 | + " g (grab - bots randomly click prims whether set clickable or not)\n" | ||
168 | + " n (none - bots do nothing)\n" | ||
169 | + " t (teleport - bots regularly teleport between regions on the grid)\n" | ||
170 | // " c (cross)\n" + | ||
171 | + " -wear folder from which to load appearance data, \"no\" if there is no such folder (default: no) (optional)\n" | ||
172 | + " -h, -help show this message.\n"); | ||
173 | } | ||
174 | } | ||
175 | } | ||