Scene.Inventory.cs 114 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767
  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. using System;
  28. using System.Collections.Generic;
  29. using System.Reflection;
  30. using System.Text;
  31. using System.Timers;
  32. using OpenMetaverse;
  33. using OpenMetaverse.Packets;
  34. using log4net;
  35. using OpenSim.Framework;
  36. using OpenSim.Framework.Communications.Cache;
  37. using OpenSim.Region.Framework;
  38. using OpenSim.Region.Framework.Interfaces;
  39. using OpenSim.Region.Framework.Scenes.Serialization;
  40. namespace OpenSim.Region.Framework.Scenes
  41. {
  42. public partial class Scene
  43. {
  44. private static readonly ILog m_log
  45. = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
  46. /// <summary>
  47. /// Allows asynchronous derezzing of objects from the scene into a client's inventory.
  48. /// </summary>
  49. protected AsyncSceneObjectGroupDeleter m_asyncSceneObjectDeleter;
  50. /// <summary>
  51. /// Start all the scripts in the scene which should be started.
  52. /// </summary>
  53. public void CreateScriptInstances()
  54. {
  55. m_log.Info("[PRIM INVENTORY]: Starting scripts in scene");
  56. foreach (EntityBase group in Entities)
  57. {
  58. if (group is SceneObjectGroup)
  59. {
  60. ((SceneObjectGroup) group).CreateScriptInstances(0, false, DefaultScriptEngine, 0);
  61. }
  62. }
  63. }
  64. public void AddUploadedInventoryItem(UUID agentID, InventoryItemBase item)
  65. {
  66. IMoneyModule money=RequestModuleInterface<IMoneyModule>();
  67. if (money != null)
  68. {
  69. money.ApplyUploadCharge(agentID);
  70. }
  71. AddInventoryItem(agentID, item);
  72. }
  73. public bool AddInventoryItemReturned(UUID AgentId, InventoryItemBase item)
  74. {
  75. CachedUserInfo userInfo
  76. = CommsManager.UserProfileCacheService.GetUserDetails(AgentId);
  77. if (userInfo != null)
  78. {
  79. userInfo.AddItem(item);
  80. return true;
  81. }
  82. else
  83. {
  84. m_log.ErrorFormat(
  85. "[AGENT INVENTORY]: Agent was not found for add of item {1} {2}", item.Name, item.ID);
  86. return false;
  87. }
  88. }
  89. public void AddInventoryItem(UUID AgentID, InventoryItemBase item)
  90. {
  91. CachedUserInfo userInfo
  92. = CommsManager.UserProfileCacheService.GetUserDetails(AgentID);
  93. if (userInfo != null)
  94. {
  95. userInfo.AddItem(item);
  96. int userlevel = 0;
  97. if (Permissions.IsGod(AgentID))
  98. {
  99. userlevel = 1;
  100. }
  101. // TODO: remove this cruft once MasterAvatar is fully deprecated
  102. //
  103. if (m_regInfo.MasterAvatarAssignedUUID == AgentID)
  104. {
  105. userlevel = 2;
  106. }
  107. EventManager.TriggerOnNewInventoryItemUploadComplete(AgentID, item.AssetID, item.Name, userlevel);
  108. }
  109. else
  110. {
  111. m_log.ErrorFormat(
  112. "[AGENT INVENTORY]: Agent {1} was not found for add of item {2} {3}",
  113. AgentID, item.Name, item.ID);
  114. return;
  115. }
  116. }
  117. /// <summary>
  118. /// Add an inventory item to an avatar's inventory.
  119. /// </summary>
  120. /// <param name="remoteClient">The remote client controlling the avatar</param>
  121. /// <param name="item">The item. This structure contains all the item metadata, including the folder
  122. /// in which the item is to be placed.</param>
  123. public void AddInventoryItem(IClientAPI remoteClient, InventoryItemBase item)
  124. {
  125. CachedUserInfo userInfo
  126. = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  127. if (userInfo != null)
  128. {
  129. AddInventoryItem(remoteClient.AgentId, item);
  130. remoteClient.SendInventoryItemCreateUpdate(item, 0);
  131. }
  132. else
  133. {
  134. m_log.ErrorFormat(
  135. "[AGENT INVENTORY]: Could not resolve user {0} for adding an inventory item",
  136. remoteClient.AgentId);
  137. }
  138. }
  139. /// <summary>
  140. /// Capability originating call to update the asset of an item in an agent's inventory
  141. /// </summary>
  142. /// <param name="remoteClient"></param>
  143. /// <param name="itemID"></param>
  144. /// <param name="data"></param>
  145. /// <returns></returns>
  146. public virtual UUID CapsUpdateInventoryItemAsset(IClientAPI remoteClient, UUID itemID, byte[] data)
  147. {
  148. CachedUserInfo userInfo = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  149. if (userInfo != null)
  150. {
  151. if (userInfo.RootFolder != null)
  152. {
  153. InventoryItemBase item = userInfo.RootFolder.FindItem(itemID);
  154. if (item != null)
  155. {
  156. if ((InventoryType)item.InvType == InventoryType.Notecard)
  157. {
  158. if (!Permissions.CanEditNotecard(itemID, UUID.Zero, remoteClient.AgentId))
  159. {
  160. remoteClient.SendAgentAlertMessage("Insufficient permissions to edit notecard", false);
  161. return UUID.Zero;
  162. }
  163. remoteClient.SendAgentAlertMessage("Notecard saved", false);
  164. }
  165. else if ((InventoryType)item.InvType == InventoryType.LSL)
  166. {
  167. if (!Permissions.CanEditScript(itemID, UUID.Zero, remoteClient.AgentId))
  168. {
  169. remoteClient.SendAgentAlertMessage("Insufficient permissions to edit script", false);
  170. return UUID.Zero;
  171. }
  172. remoteClient.SendAgentAlertMessage("Script saved", false);
  173. }
  174. AssetBase asset =
  175. CreateAsset(item.Name, item.Description, (sbyte)item.AssetType, data);
  176. item.AssetID = asset.FullID;
  177. AssetService.Store(asset);
  178. userInfo.UpdateItem(item);
  179. // remoteClient.SendInventoryItemCreateUpdate(item);
  180. return (asset.FullID);
  181. }
  182. }
  183. }
  184. else
  185. {
  186. m_log.ErrorFormat(
  187. "[AGENT INVENTORY]: Could not resolve user {0} for caps inventory update",
  188. remoteClient.AgentId);
  189. }
  190. return UUID.Zero;
  191. }
  192. /// <summary>
  193. /// <see>CapsUpdatedInventoryItemAsset(IClientAPI, UUID, byte[])</see>
  194. /// </summary>
  195. public UUID CapsUpdateInventoryItemAsset(UUID avatarId, UUID itemID, byte[] data)
  196. {
  197. ScenePresence avatar;
  198. if (TryGetAvatar(avatarId, out avatar))
  199. {
  200. return CapsUpdateInventoryItemAsset(avatar.ControllingClient, itemID, data);
  201. }
  202. else
  203. {
  204. m_log.ErrorFormat(
  205. "[AGENT INVENTORY]: " +
  206. "Avatar {0} cannot be found to update its inventory item asset",
  207. avatarId);
  208. }
  209. return UUID.Zero;
  210. }
  211. /// <summary>
  212. /// Capability originating call to update the asset of a script in a prim's (task's) inventory
  213. /// </summary>
  214. /// <param name="remoteClient"></param>
  215. /// <param name="itemID"></param>
  216. /// <param name="primID">The prim which contains the item to update</param>
  217. /// <param name="isScriptRunning">Indicates whether the script to update is currently running</param>
  218. /// <param name="data"></param>
  219. public void CapsUpdateTaskInventoryScriptAsset(IClientAPI remoteClient, UUID itemId,
  220. UUID primId, bool isScriptRunning, byte[] data)
  221. {
  222. if (!Permissions.CanEditScript(itemId, primId, remoteClient.AgentId))
  223. {
  224. remoteClient.SendAgentAlertMessage("Insufficient permissions to edit script", false);
  225. return;
  226. }
  227. // Retrieve group
  228. SceneObjectPart part = GetSceneObjectPart(primId);
  229. SceneObjectGroup group = part.ParentGroup;
  230. if (null == group)
  231. {
  232. m_log.ErrorFormat(
  233. "[PRIM INVENTORY]: " +
  234. "Prim inventory update requested for item ID {0} in prim ID {1} but this prim does not exist",
  235. itemId, primId);
  236. return;
  237. }
  238. // Retrieve item
  239. TaskInventoryItem item = group.GetInventoryItem(part.LocalId, itemId);
  240. if (null == item)
  241. {
  242. m_log.ErrorFormat(
  243. "[PRIM INVENTORY]: Tried to retrieve item ID {0} from prim {1}, {2} for caps script update "
  244. + " but the item does not exist in this inventory",
  245. itemId, part.Name, part.UUID);
  246. return;
  247. }
  248. AssetBase asset = CreateAsset(item.Name, item.Description, (sbyte)AssetType.LSLText, data);
  249. AssetService.Store(asset);
  250. if (isScriptRunning)
  251. {
  252. part.Inventory.RemoveScriptInstance(item.ItemID);
  253. }
  254. // Update item with new asset
  255. item.AssetID = asset.FullID;
  256. group.UpdateInventoryItem(item);
  257. part.GetProperties(remoteClient);
  258. // Trigger rerunning of script (use TriggerRezScript event, see RezScript)
  259. if (isScriptRunning)
  260. {
  261. // Needs to determine which engine was running it and use that
  262. //
  263. part.Inventory.CreateScriptInstance(item.ItemID, 0, false, DefaultScriptEngine, 0);
  264. }
  265. else
  266. {
  267. remoteClient.SendAgentAlertMessage("Script saved", false);
  268. }
  269. }
  270. /// <summary>
  271. /// <see>CapsUpdateTaskInventoryScriptAsset(IClientAPI, UUID, UUID, bool, byte[])</see>
  272. /// </summary>
  273. public void CapsUpdateTaskInventoryScriptAsset(UUID avatarId, UUID itemId,
  274. UUID primId, bool isScriptRunning, byte[] data)
  275. {
  276. ScenePresence avatar;
  277. if (TryGetAvatar(avatarId, out avatar))
  278. {
  279. CapsUpdateTaskInventoryScriptAsset(
  280. avatar.ControllingClient, itemId, primId, isScriptRunning, data);
  281. }
  282. else
  283. {
  284. m_log.ErrorFormat(
  285. "[PRIM INVENTORY]: " +
  286. "Avatar {0} cannot be found to update its prim item asset",
  287. avatarId);
  288. }
  289. }
  290. /// <summary>
  291. /// Update an item which is either already in the client's inventory or is within
  292. /// a transaction
  293. /// </summary>
  294. /// <param name="remoteClient"></param>
  295. /// <param name="transactionID">The transaction ID. If this is UUID.Zero we will
  296. /// assume that we are not in a transaction</param>
  297. /// <param name="itemID">The ID of the updated item</param>
  298. /// <param name="name">The name of the updated item</param>
  299. /// <param name="description">The description of the updated item</param>
  300. /// <param name="nextOwnerMask">The permissions of the updated item</param>
  301. /* public void UpdateInventoryItemAsset(IClientAPI remoteClient, UUID transactionID,
  302. UUID itemID, string name, string description,
  303. uint nextOwnerMask)*/
  304. public void UpdateInventoryItemAsset(IClientAPI remoteClient, UUID transactionID,
  305. UUID itemID, InventoryItemBase itemUpd)
  306. {
  307. CachedUserInfo userInfo
  308. = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  309. if (userInfo != null && userInfo.RootFolder != null)
  310. {
  311. InventoryItemBase item = userInfo.RootFolder.FindItem(itemID);
  312. if (item != null)
  313. {
  314. if (UUID.Zero == transactionID)
  315. {
  316. item.Name = itemUpd.Name;
  317. item.Description = itemUpd.Description;
  318. item.NextPermissions = itemUpd.NextPermissions;
  319. item.CurrentPermissions |= 8; // Slam!
  320. item.EveryOnePermissions = itemUpd.EveryOnePermissions;
  321. item.GroupPermissions = itemUpd.GroupPermissions;
  322. item.GroupID = itemUpd.GroupID;
  323. item.GroupOwned = itemUpd.GroupOwned;
  324. item.CreationDate = itemUpd.CreationDate;
  325. // The client sends zero if its newly created?
  326. if (itemUpd.CreationDate == 0)
  327. item.CreationDate = Util.UnixTimeSinceEpoch();
  328. else
  329. item.CreationDate = itemUpd.CreationDate;
  330. // TODO: Check if folder changed and move item
  331. //item.NextPermissions = itemUpd.Folder;
  332. item.InvType = itemUpd.InvType;
  333. item.SalePrice = itemUpd.SalePrice;
  334. item.SaleType = itemUpd.SaleType;
  335. item.Flags = itemUpd.Flags;
  336. userInfo.UpdateItem(item);
  337. }
  338. else
  339. {
  340. IAgentAssetTransactions agentTransactions = this.RequestModuleInterface<IAgentAssetTransactions>();
  341. if (agentTransactions != null)
  342. {
  343. agentTransactions.HandleItemUpdateFromTransaction(
  344. remoteClient, transactionID, item);
  345. }
  346. }
  347. }
  348. else
  349. {
  350. m_log.Error(
  351. "[AGENTINVENTORY]: Item ID " + itemID + " not found for an inventory item update.");
  352. }
  353. }
  354. else
  355. {
  356. m_log.Error(
  357. "[AGENT INVENTORY]: Agent ID " + remoteClient.AgentId + " not found for an inventory item update.");
  358. }
  359. }
  360. /// <summary>
  361. /// Give an inventory item from one user to another
  362. /// </summary>
  363. /// <param name="recipientClient"></param>
  364. /// <param name="senderId">ID of the sender of the item</param>
  365. /// <param name="itemId"></param>
  366. public virtual void GiveInventoryItem(IClientAPI recipientClient, UUID senderId, UUID itemId)
  367. {
  368. InventoryItemBase itemCopy = GiveInventoryItem(recipientClient.AgentId, senderId, itemId);
  369. if (itemCopy != null)
  370. recipientClient.SendBulkUpdateInventory(itemCopy);
  371. }
  372. /// <summary>
  373. /// Give an inventory item from one user to another
  374. /// </summary>
  375. /// <param name="recipient"></param>
  376. /// <param name="senderId">ID of the sender of the item</param>
  377. /// <param name="itemId"></param>
  378. /// <returns>The inventory item copy given, null if the give was unsuccessful</returns>
  379. public virtual InventoryItemBase GiveInventoryItem(UUID recipient, UUID senderId, UUID itemId)
  380. {
  381. return GiveInventoryItem(recipient, senderId, itemId, UUID.Zero);
  382. }
  383. /// <summary>
  384. /// Give an inventory item from one user to another
  385. /// </summary>
  386. /// <param name="recipient"></param>
  387. /// <param name="senderId">ID of the sender of the item</param>
  388. /// <param name="itemId"></param>
  389. /// <param name="recipientFolderId">
  390. /// The id of the folder in which the copy item should go. If UUID.Zero then the item is placed in the most
  391. /// appropriate default folder.
  392. /// </param>
  393. /// <returns>
  394. /// The inventory item copy given, null if the give was unsuccessful
  395. /// </returns>
  396. public virtual InventoryItemBase GiveInventoryItem(
  397. UUID recipient, UUID senderId, UUID itemId, UUID recipientFolderId)
  398. {
  399. // Retrieve the item from the sender
  400. CachedUserInfo senderUserInfo = CommsManager.UserProfileCacheService.GetUserDetails(senderId);
  401. Console.WriteLine("Scene.Inventory.cs: GiveInventoryItem");
  402. if (senderUserInfo == null)
  403. {
  404. m_log.ErrorFormat(
  405. "[AGENT INVENTORY]: Failed to find sending user {0} for item {1}", senderId, itemId);
  406. return null;
  407. }
  408. if (senderUserInfo.RootFolder != null)
  409. {
  410. InventoryItemBase item = senderUserInfo.RootFolder.FindItem(itemId);
  411. if (item != null)
  412. {
  413. if (!Permissions.BypassPermissions())
  414. {
  415. if ((item.CurrentPermissions & (uint)PermissionMask.Transfer) == 0)
  416. return null;
  417. }
  418. CachedUserInfo recipientUserInfo
  419. = CommsManager.UserProfileCacheService.GetUserDetails(recipient);
  420. if (recipientUserInfo != null)
  421. {
  422. if (!recipientUserInfo.HasReceivedInventory)
  423. recipientUserInfo.FetchInventory();
  424. // Insert a copy of the item into the recipient
  425. InventoryItemBase itemCopy = new InventoryItemBase();
  426. itemCopy.Owner = recipient;
  427. itemCopy.CreatorId = item.CreatorId;
  428. itemCopy.ID = UUID.Random();
  429. itemCopy.AssetID = item.AssetID;
  430. itemCopy.Description = item.Description;
  431. itemCopy.Name = item.Name;
  432. itemCopy.AssetType = item.AssetType;
  433. itemCopy.InvType = item.InvType;
  434. itemCopy.Folder = recipientFolderId;
  435. if (Permissions.PropagatePermissions())
  436. {
  437. if (item.InvType == (int)InventoryType.Object)
  438. {
  439. itemCopy.BasePermissions &= ~(uint)(PermissionMask.Copy | PermissionMask.Modify | PermissionMask.Transfer);
  440. itemCopy.BasePermissions |= (item.CurrentPermissions & 7) << 13;
  441. }
  442. else
  443. {
  444. itemCopy.BasePermissions = item.BasePermissions & item.NextPermissions;
  445. }
  446. itemCopy.CurrentPermissions = itemCopy.BasePermissions;
  447. if ((item.CurrentPermissions & 8) != 0) // Propagate slam bit
  448. {
  449. itemCopy.BasePermissions &= item.NextPermissions;
  450. itemCopy.CurrentPermissions = itemCopy.BasePermissions;
  451. itemCopy.CurrentPermissions |= 8;
  452. }
  453. itemCopy.NextPermissions = item.NextPermissions;
  454. itemCopy.EveryOnePermissions = item.EveryOnePermissions & item.NextPermissions;
  455. itemCopy.GroupPermissions = item.GroupPermissions & item.NextPermissions;
  456. }
  457. else
  458. {
  459. itemCopy.CurrentPermissions = item.CurrentPermissions;
  460. itemCopy.NextPermissions = item.NextPermissions;
  461. itemCopy.EveryOnePermissions = item.EveryOnePermissions & item.NextPermissions;
  462. itemCopy.GroupPermissions = item.GroupPermissions & item.NextPermissions;
  463. itemCopy.BasePermissions = item.BasePermissions;
  464. }
  465. itemCopy.GroupID = UUID.Zero;
  466. itemCopy.GroupOwned = false;
  467. itemCopy.Flags = item.Flags;
  468. itemCopy.SalePrice = item.SalePrice;
  469. itemCopy.SaleType = item.SaleType;
  470. recipientUserInfo.AddItem(itemCopy);
  471. if (!Permissions.BypassPermissions())
  472. {
  473. if ((item.CurrentPermissions & (uint)PermissionMask.Copy) == 0)
  474. senderUserInfo.DeleteItem(itemId);
  475. }
  476. return itemCopy;
  477. }
  478. else
  479. {
  480. m_log.ErrorFormat(
  481. "[AGENT INVENTORY]: Could not find userinfo for recipient user {0} of item {1}, {2} from {3}",
  482. recipient, item.Name,
  483. item.ID, senderId);
  484. }
  485. }
  486. else
  487. {
  488. m_log.ErrorFormat(
  489. "[AGENT INVENTORY]: Failed to find item {0} to give to {1}", itemId, senderId);
  490. return null;
  491. }
  492. }
  493. else
  494. {
  495. m_log.Error("[AGENT INVENTORY]: Failed to find item " + itemId.ToString() + ", no root folder");
  496. return null;
  497. }
  498. return null;
  499. }
  500. /// <summary>
  501. /// Give an entire inventory folder from one user to another. The entire contents (including all descendent
  502. /// folders) is given.
  503. /// </summary>
  504. /// <param name="recipientId"></param>
  505. /// <param name="senderId">ID of the sender of the item</param>
  506. /// <param name="folderId"></param>
  507. /// <param name="recipientParentFolderId">
  508. /// The id of the receipient folder in which the send folder should be placed. If UUID.Zero then the
  509. /// recipient folder is the root folder
  510. /// </param>
  511. /// <returns>
  512. /// The inventory folder copy given, null if the copy was unsuccessful
  513. /// </returns>
  514. public virtual InventoryFolderImpl GiveInventoryFolder(
  515. UUID recipientId, UUID senderId, UUID folderId, UUID recipientParentFolderId)
  516. {
  517. // Retrieve the folder from the sender
  518. CachedUserInfo senderUserInfo = CommsManager.UserProfileCacheService.GetUserDetails(senderId);
  519. if (null == senderUserInfo)
  520. {
  521. m_log.ErrorFormat(
  522. "[AGENT INVENTORY]: Failed to find sending user {0} for folder {1}", senderId, folderId);
  523. return null;
  524. }
  525. if (!senderUserInfo.HasReceivedInventory)
  526. {
  527. m_log.DebugFormat(
  528. "[AGENT INVENTORY]: Could not give inventory folder - have not yet received inventory for {0}",
  529. senderId);
  530. return null;
  531. }
  532. InventoryFolderImpl folder = senderUserInfo.RootFolder.FindFolder(folderId);
  533. if (null == folder)
  534. {
  535. m_log.ErrorFormat(
  536. "[AGENT INVENTORY]: Could not find inventory folder {0} to give", folderId);
  537. return null;
  538. }
  539. CachedUserInfo recipientUserInfo
  540. = CommsManager.UserProfileCacheService.GetUserDetails(recipientId);
  541. if (null == recipientUserInfo)
  542. {
  543. m_log.ErrorFormat(
  544. "[AGENT INVENTORY]: Failed to find receiving user {0} for folder {1}", recipientId, folderId);
  545. return null;
  546. }
  547. if (!recipientUserInfo.HasReceivedInventory)
  548. {
  549. recipientUserInfo.FetchInventory();
  550. if (!WaitForInventory(recipientUserInfo))
  551. return null;
  552. }
  553. if (recipientParentFolderId == UUID.Zero)
  554. recipientParentFolderId = recipientUserInfo.RootFolder.ID;
  555. UUID newFolderId = UUID.Random();
  556. recipientUserInfo.CreateFolder(folder.Name, newFolderId, (ushort)folder.Type, recipientParentFolderId);
  557. // XXX: Messy - we should really get this back in the CreateFolder call
  558. InventoryFolderImpl copiedFolder = recipientUserInfo.RootFolder.FindFolder(newFolderId);
  559. // Give all the subfolders
  560. List<InventoryFolderImpl> subFolders = folder.RequestListOfFolderImpls();
  561. foreach (InventoryFolderImpl childFolder in subFolders)
  562. {
  563. GiveInventoryFolder(recipientId, senderId, childFolder.ID, copiedFolder.ID);
  564. }
  565. // Give all the items
  566. List<InventoryItemBase> items = folder.RequestListOfItems();
  567. foreach (InventoryItemBase item in items)
  568. {
  569. GiveInventoryItem(recipientId, senderId, item.ID, copiedFolder.ID);
  570. }
  571. return copiedFolder;
  572. }
  573. public void CopyInventoryItem(IClientAPI remoteClient, uint callbackID, UUID oldAgentID, UUID oldItemID,
  574. UUID newFolderID, string newName)
  575. {
  576. m_log.DebugFormat(
  577. "[AGENT INVENTORY]: CopyInventoryItem received by {0} with oldAgentID {1}, oldItemID {2}, new FolderID {3}, newName {4}",
  578. remoteClient.AgentId, oldAgentID, oldItemID, newFolderID, newName);
  579. InventoryItemBase item = CommsManager.UserProfileCacheService.LibraryRoot.FindItem(oldItemID);
  580. if (item == null)
  581. {
  582. CachedUserInfo userInfo = CommsManager.UserProfileCacheService.GetUserDetails(oldAgentID);
  583. if (userInfo == null)
  584. {
  585. m_log.Error("[AGENT INVENTORY]: Failed to find user " + oldAgentID.ToString());
  586. return;
  587. }
  588. if (userInfo.RootFolder != null)
  589. {
  590. item = userInfo.RootFolder.FindItem(oldItemID);
  591. if (item == null)
  592. {
  593. m_log.Error("[AGENT INVENTORY]: Failed to find item " + oldItemID.ToString());
  594. return;
  595. }
  596. }
  597. else
  598. {
  599. m_log.Error("[AGENT INVENTORY]: Failed to find item " + oldItemID.ToString());
  600. return;
  601. }
  602. }
  603. AssetBase asset = AssetService.Get(item.AssetID.ToString());
  604. if (asset != null)
  605. {
  606. if (newName != String.Empty)
  607. {
  608. asset.Name = newName;
  609. }
  610. else
  611. {
  612. newName = item.Name;
  613. }
  614. if (remoteClient.AgentId == oldAgentID)
  615. {
  616. CreateNewInventoryItem(
  617. remoteClient, item.CreatorId, newFolderID, newName, item.Flags, callbackID, asset, (sbyte)item.InvType,
  618. item.BasePermissions, item.CurrentPermissions, item.EveryOnePermissions, item.NextPermissions, item.GroupPermissions, Util.UnixTimeSinceEpoch());
  619. }
  620. else
  621. {
  622. CreateNewInventoryItem(
  623. remoteClient, item.CreatorId, newFolderID, newName, item.Flags, callbackID, asset, (sbyte)item.InvType,
  624. item.NextPermissions, item.NextPermissions, item.EveryOnePermissions & item.NextPermissions, item.NextPermissions, item.GroupPermissions, Util.UnixTimeSinceEpoch());
  625. }
  626. }
  627. else
  628. {
  629. m_log.ErrorFormat(
  630. "[AGENT INVENTORY]: Could not copy item {0} since asset {1} could not be found",
  631. item.Name, item.AssetID);
  632. }
  633. }
  634. /// <summary>
  635. /// Create a new asset data structure.
  636. /// </summary>
  637. /// <param name="name"></param>
  638. /// <param name="description"></param>
  639. /// <param name="invType"></param>
  640. /// <param name="assetType"></param>
  641. /// <param name="data"></param>
  642. /// <returns></returns>
  643. private AssetBase CreateAsset(string name, string description, sbyte assetType, byte[] data)
  644. {
  645. AssetBase asset = new AssetBase();
  646. asset.Name = name;
  647. asset.Description = description;
  648. asset.Type = assetType;
  649. asset.FullID = UUID.Random();
  650. asset.Data = (data == null) ? new byte[1] : data;
  651. return asset;
  652. }
  653. /// <summary>
  654. /// Move an item within the agent's inventory.
  655. /// </summary>
  656. /// <param name="remoteClient"></param>
  657. /// <param name="folderID"></param>
  658. /// <param name="itemID"></param>
  659. /// <param name="length"></param>
  660. /// <param name="newName"></param>
  661. public void MoveInventoryItem(IClientAPI remoteClient, UUID folderID, UUID itemID, int length,
  662. string newName)
  663. {
  664. m_log.DebugFormat(
  665. "[AGENT INVENTORY]: Moving item {0} to {1} for {2}", itemID, folderID, remoteClient.AgentId);
  666. CachedUserInfo userInfo = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  667. if (userInfo == null)
  668. {
  669. m_log.Error("[AGENT INVENTORY]: Failed to find user " + remoteClient.AgentId.ToString());
  670. return;
  671. }
  672. if (userInfo.RootFolder != null)
  673. {
  674. InventoryItemBase item = userInfo.RootFolder.FindItem(itemID);
  675. if (item != null)
  676. {
  677. if (newName != String.Empty)
  678. {
  679. item.Name = newName;
  680. }
  681. item.Folder = folderID;
  682. userInfo.DeleteItem(item.ID);
  683. AddInventoryItem(remoteClient, item);
  684. }
  685. else
  686. {
  687. m_log.Error("[AGENT INVENTORY]: Failed to find item " + itemID.ToString());
  688. return;
  689. }
  690. }
  691. else
  692. {
  693. m_log.Error("[AGENT INVENTORY]: Failed to find item " + itemID.ToString() + ", no root folder");
  694. return;
  695. }
  696. }
  697. /// <summary>
  698. /// Create a new inventory item.
  699. /// </summary>
  700. /// <param name="remoteClient"></param>
  701. /// <param name="folderID"></param>
  702. /// <param name="callbackID"></param>
  703. /// <param name="asset"></param>
  704. /// <param name="invType"></param>
  705. /// <param name="nextOwnerMask"></param>
  706. private void CreateNewInventoryItem(IClientAPI remoteClient, string creatorID, UUID folderID, string name, uint flags, uint callbackID,
  707. AssetBase asset, sbyte invType, uint nextOwnerMask, int creationDate)
  708. {
  709. CreateNewInventoryItem(
  710. remoteClient, creatorID, folderID, name, flags, callbackID, asset, invType,
  711. (uint)PermissionMask.All, (uint)PermissionMask.All, 0, nextOwnerMask, 0, creationDate);
  712. }
  713. /// <summary>
  714. /// Create a new Inventory Item
  715. /// </summary>
  716. /// <param name="remoteClient"></param>
  717. /// <param name="folderID"></param>
  718. /// <param name="callbackID"></param>
  719. /// <param name="asset"></param>
  720. /// <param name="invType"></param>
  721. /// <param name="nextOwnerMask"></param>
  722. /// <param name="creationDate"></param>
  723. private void CreateNewInventoryItem(
  724. IClientAPI remoteClient, string creatorID, UUID folderID, string name, uint flags, uint callbackID, AssetBase asset, sbyte invType,
  725. uint baseMask, uint currentMask, uint everyoneMask, uint nextOwnerMask, uint groupMask, int creationDate)
  726. {
  727. CachedUserInfo userInfo
  728. = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  729. if (userInfo != null)
  730. {
  731. InventoryItemBase item = new InventoryItemBase();
  732. item.Owner = remoteClient.AgentId;
  733. item.CreatorId = creatorID;
  734. item.ID = UUID.Random();
  735. item.AssetID = asset.FullID;
  736. item.Description = asset.Description;
  737. item.Name = name;
  738. item.Flags = flags;
  739. item.AssetType = asset.Type;
  740. item.InvType = invType;
  741. item.Folder = folderID;
  742. item.CurrentPermissions = currentMask;
  743. item.NextPermissions = nextOwnerMask;
  744. item.EveryOnePermissions = everyoneMask;
  745. item.GroupPermissions = groupMask;
  746. item.BasePermissions = baseMask;
  747. item.CreationDate = creationDate;
  748. userInfo.AddItem(item);
  749. remoteClient.SendInventoryItemCreateUpdate(item, callbackID);
  750. }
  751. else
  752. {
  753. m_log.WarnFormat(
  754. "No user details associated with client {0} uuid {1} in CreateNewInventoryItem!",
  755. remoteClient.Name, remoteClient.AgentId);
  756. }
  757. }
  758. /// <summary>
  759. /// Create a new inventory item. Called when the client creates a new item directly within their
  760. /// inventory (e.g. by selecting a context inventory menu option).
  761. /// </summary>
  762. /// <param name="remoteClient"></param>
  763. /// <param name="transactionID"></param>
  764. /// <param name="folderID"></param>
  765. /// <param name="callbackID"></param>
  766. /// <param name="description"></param>
  767. /// <param name="name"></param>
  768. /// <param name="invType"></param>
  769. /// <param name="type"></param>
  770. /// <param name="wearableType"></param>
  771. /// <param name="nextOwnerMask"></param>
  772. public void CreateNewInventoryItem(IClientAPI remoteClient, UUID transactionID, UUID folderID,
  773. uint callbackID, string description, string name, sbyte invType,
  774. sbyte assetType,
  775. byte wearableType, uint nextOwnerMask, int creationDate)
  776. {
  777. m_log.DebugFormat("[AGENT INVENTORY]: Received request to create inventory item {0} in folder {1}", name, folderID);
  778. if (!Permissions.CanCreateUserInventory(invType, remoteClient.AgentId))
  779. return;
  780. if (transactionID == UUID.Zero)
  781. {
  782. CachedUserInfo userInfo
  783. = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  784. if (userInfo != null)
  785. {
  786. ScenePresence presence;
  787. TryGetAvatar(remoteClient.AgentId, out presence);
  788. byte[] data = null;
  789. if (invType == (sbyte)InventoryType.Landmark && presence != null)
  790. {
  791. Vector3 pos = presence.AbsolutePosition;
  792. string strdata = String.Format(
  793. "Landmark version 2\nregion_id {0}\nlocal_pos {1} {2} {3}\nregion_handle {4}\n",
  794. presence.Scene.RegionInfo.RegionID,
  795. pos.X, pos.Y, pos.Z,
  796. presence.RegionHandle);
  797. data = Encoding.ASCII.GetBytes(strdata);
  798. }
  799. AssetBase asset = CreateAsset(name, description, assetType, data);
  800. AssetService.Store(asset);
  801. CreateNewInventoryItem(remoteClient, remoteClient.AgentId.ToString(), folderID, asset.Name, 0, callbackID, asset, invType, nextOwnerMask, creationDate);
  802. }
  803. else
  804. {
  805. m_log.ErrorFormat(
  806. "userInfo for agent uuid {0} unexpectedly null in CreateNewInventoryItem",
  807. remoteClient.AgentId);
  808. }
  809. }
  810. else
  811. {
  812. IAgentAssetTransactions agentTransactions = this.RequestModuleInterface<IAgentAssetTransactions>();
  813. if (agentTransactions != null)
  814. {
  815. agentTransactions.HandleItemCreationFromTransaction(
  816. remoteClient, transactionID, folderID, callbackID, description,
  817. name, invType, assetType, wearableType, nextOwnerMask);
  818. }
  819. }
  820. }
  821. /// <summary>
  822. /// Remove an inventory item for the client's inventory
  823. /// </summary>
  824. /// <param name="remoteClient"></param>
  825. /// <param name="itemID"></param>
  826. private void RemoveInventoryItem(IClientAPI remoteClient, UUID itemID)
  827. {
  828. CachedUserInfo userInfo
  829. = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  830. if (userInfo == null)
  831. {
  832. m_log.WarnFormat(
  833. "[AGENT INVENTORY]: Failed to find user {0} {1} to delete inventory item {2}",
  834. remoteClient.Name, remoteClient.AgentId, itemID);
  835. return;
  836. }
  837. userInfo.DeleteItem(itemID);
  838. }
  839. /// <summary>
  840. /// Removes an inventory folder. Although there is a packet in the Linden protocol for this, it may be
  841. /// legacy and not currently used (purge folder is used to remove folders from trash instead).
  842. /// </summary>
  843. /// <param name="remoteClient"></param>
  844. /// <param name="folderID"></param>
  845. private void RemoveInventoryFolder(IClientAPI remoteClient, UUID folderID)
  846. {
  847. CachedUserInfo userInfo
  848. = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  849. if (userInfo == null)
  850. {
  851. m_log.Warn("[AGENT INVENTORY]: Failed to find user " + remoteClient.AgentId.ToString());
  852. return;
  853. }
  854. if (userInfo.RootFolder != null)
  855. {
  856. InventoryItemBase folder = userInfo.RootFolder.FindItem(folderID);
  857. if (folder != null)
  858. {
  859. m_log.WarnFormat(
  860. "[AGENT INVENTORY]: Remove folder not implemented in request by {0} {1} for {2}",
  861. remoteClient.Name, remoteClient.AgentId, folderID);
  862. // doesn't work just yet, commented out. will fix in next patch.
  863. // userInfo.DeleteItem(folder);
  864. }
  865. }
  866. }
  867. private SceneObjectGroup GetGroupByPrim(uint localID)
  868. {
  869. List<EntityBase> EntityList = GetEntities();
  870. foreach (EntityBase ent in EntityList)
  871. {
  872. if (ent is SceneObjectGroup)
  873. {
  874. if (((SceneObjectGroup) ent).HasChildPrim(localID))
  875. return (SceneObjectGroup) ent;
  876. }
  877. }
  878. return null;
  879. }
  880. /// <summary>
  881. /// Send the details of a prim's inventory to the client.
  882. /// </summary>
  883. /// <param name="remoteClient"></param>
  884. /// <param name="primLocalID"></param>
  885. public void RequestTaskInventory(IClientAPI remoteClient, uint primLocalID)
  886. {
  887. SceneObjectGroup group = GetGroupByPrim(primLocalID);
  888. if (group != null)
  889. {
  890. bool fileChange = group.GetPartInventoryFileName(remoteClient, primLocalID);
  891. if (fileChange)
  892. {
  893. if (XferManager != null)
  894. {
  895. group.RequestInventoryFile(remoteClient, primLocalID, XferManager);
  896. }
  897. }
  898. }
  899. else
  900. {
  901. m_log.ErrorFormat(
  902. "[PRIM INVENTORY]: Inventory requested of prim {0} which doesn't exist", primLocalID);
  903. }
  904. }
  905. /// <summary>
  906. /// Remove an item from a prim (task) inventory
  907. /// </summary>
  908. /// <param name="remoteClient">Unused at the moment but retained since the avatar ID might
  909. /// be necessary for a permissions check at some stage.</param>
  910. /// <param name="itemID"></param>
  911. /// <param name="localID"></param>
  912. public void RemoveTaskInventory(IClientAPI remoteClient, UUID itemID, uint localID)
  913. {
  914. SceneObjectPart part = GetSceneObjectPart(localID);
  915. SceneObjectGroup group = part.ParentGroup;
  916. if (group != null)
  917. {
  918. TaskInventoryItem item = group.GetInventoryItem(localID, itemID);
  919. if (item == null)
  920. return;
  921. if (item.Type == 10)
  922. {
  923. EventManager.TriggerRemoveScript(localID, itemID);
  924. }
  925. group.RemoveInventoryItem(localID, itemID);
  926. part.GetProperties(remoteClient);
  927. }
  928. else
  929. {
  930. m_log.ErrorFormat(
  931. "[PRIM INVENTORY]: " +
  932. "Removal of item {0} requested of prim {1} but this prim does not exist",
  933. itemID,
  934. localID);
  935. }
  936. }
  937. private InventoryItemBase CreateAgentInventoryItemFromTask(UUID destAgent, SceneObjectPart part, UUID itemId)
  938. {
  939. Console.WriteLine("CreateAgentInventoryItemFromTask");
  940. TaskInventoryItem taskItem = part.Inventory.GetInventoryItem(itemId);
  941. if (null == taskItem)
  942. {
  943. m_log.ErrorFormat(
  944. "[PRIM INVENTORY]: Tried to retrieve item ID {0} from prim {1}, {2} for creating an avatar"
  945. + " inventory item from a prim's inventory item "
  946. + " but the required item does not exist in the prim's inventory",
  947. itemId, part.Name, part.UUID);
  948. return null;
  949. }
  950. if ((destAgent != taskItem.OwnerID) && ((taskItem.CurrentPermissions & (uint)PermissionMask.Transfer) == 0))
  951. {
  952. return null;
  953. }
  954. InventoryItemBase agentItem = new InventoryItemBase();
  955. agentItem.ID = UUID.Random();
  956. agentItem.CreatorId = taskItem.CreatorID.ToString();
  957. agentItem.Owner = destAgent;
  958. agentItem.AssetID = taskItem.AssetID;
  959. agentItem.Description = taskItem.Description;
  960. agentItem.Name = taskItem.Name;
  961. agentItem.AssetType = taskItem.Type;
  962. agentItem.InvType = taskItem.InvType;
  963. agentItem.Flags = taskItem.Flags;
  964. if ((part.OwnerID != destAgent) && Permissions.PropagatePermissions())
  965. {
  966. if (taskItem.InvType == (int)InventoryType.Object)
  967. agentItem.BasePermissions = taskItem.BasePermissions & ((taskItem.CurrentPermissions & 7) << 13);
  968. else
  969. agentItem.BasePermissions = taskItem.BasePermissions;
  970. agentItem.BasePermissions &= taskItem.NextPermissions;
  971. agentItem.CurrentPermissions = agentItem.BasePermissions | 8;
  972. agentItem.NextPermissions = taskItem.NextPermissions;
  973. agentItem.EveryOnePermissions = taskItem.EveryonePermissions & taskItem.NextPermissions;
  974. agentItem.GroupPermissions = taskItem.GroupPermissions & taskItem.NextPermissions;
  975. }
  976. else
  977. {
  978. agentItem.BasePermissions = taskItem.BasePermissions;
  979. agentItem.CurrentPermissions = taskItem.CurrentPermissions;
  980. agentItem.NextPermissions = taskItem.NextPermissions;
  981. agentItem.EveryOnePermissions = taskItem.EveryonePermissions;
  982. agentItem.GroupPermissions = taskItem.GroupPermissions;
  983. }
  984. if (!Permissions.BypassPermissions())
  985. {
  986. if ((taskItem.CurrentPermissions & (uint)PermissionMask.Copy) == 0)
  987. part.Inventory.RemoveInventoryItem(itemId);
  988. }
  989. return agentItem;
  990. }
  991. /// <summary>
  992. /// Move the given item in the given prim to a folder in the client's inventory
  993. /// </summary>
  994. /// <param name="remoteClient"></param>
  995. /// <param name="folderID"></param>
  996. /// <param name="part"></param>
  997. /// <param name="itemID"></param>
  998. public InventoryItemBase MoveTaskInventoryItem(IClientAPI remoteClient, UUID folderId, SceneObjectPart part, UUID itemId)
  999. {
  1000. m_log.Info("Adding task inventory");
  1001. InventoryItemBase agentItem = CreateAgentInventoryItemFromTask(remoteClient.AgentId, part, itemId);
  1002. if (agentItem == null)
  1003. return null;
  1004. agentItem.Folder = folderId;
  1005. AddInventoryItem(remoteClient, agentItem);
  1006. return agentItem;
  1007. }
  1008. /// <summary>
  1009. /// <see>ClientMoveTaskInventoryItem</see>
  1010. /// </summary>
  1011. /// <param name="remoteClient"></param>
  1012. /// <param name="folderID"></param>
  1013. /// <param name="primLocalID"></param>
  1014. /// <param name="itemID"></param>
  1015. public void ClientMoveTaskInventoryItem(IClientAPI remoteClient, UUID folderId, uint primLocalId, UUID itemId)
  1016. {
  1017. SceneObjectPart part = GetSceneObjectPart(primLocalId);
  1018. if (null == part)
  1019. {
  1020. m_log.WarnFormat(
  1021. "[PRIM INVENTORY]: " +
  1022. "Move of inventory item {0} from prim with local id {1} failed because the prim could not be found",
  1023. itemId, primLocalId);
  1024. return;
  1025. }
  1026. TaskInventoryItem taskItem = part.Inventory.GetInventoryItem(itemId);
  1027. if (null == taskItem)
  1028. {
  1029. m_log.WarnFormat("[PRIM INVENTORY]: Move of inventory item {0} from prim with local id {1} failed"
  1030. + " because the inventory item could not be found",
  1031. itemId, primLocalId);
  1032. return;
  1033. }
  1034. // Only owner can copy
  1035. if (remoteClient.AgentId != taskItem.OwnerID)
  1036. return;
  1037. MoveTaskInventoryItem(remoteClient, folderId, part, itemId);
  1038. }
  1039. /// <summary>
  1040. /// <see>MoveTaskInventoryItem</see>
  1041. /// </summary>
  1042. /// <param name="remoteClient"></param>
  1043. /// <param name="folderID"></param>
  1044. /// <param name="part"></param>
  1045. /// <param name="itemID"></param>
  1046. public InventoryItemBase MoveTaskInventoryItem(UUID avatarId, UUID folderId, SceneObjectPart part, UUID itemId)
  1047. {
  1048. ScenePresence avatar;
  1049. if (TryGetAvatar(avatarId, out avatar))
  1050. {
  1051. return MoveTaskInventoryItem(avatar.ControllingClient, folderId, part, itemId);
  1052. }
  1053. else
  1054. {
  1055. CachedUserInfo profile = CommsManager.UserProfileCacheService.GetUserDetails(avatarId);
  1056. if (profile == null || profile.RootFolder == null)
  1057. {
  1058. m_log.ErrorFormat(
  1059. "[PRIM INVENTORY]: " +
  1060. "Avatar {0} cannot be found to add item",
  1061. avatarId);
  1062. return null;
  1063. }
  1064. if (!profile.HasReceivedInventory)
  1065. profile.FetchInventory();
  1066. InventoryItemBase agentItem = CreateAgentInventoryItemFromTask(avatarId, part, itemId);
  1067. if (agentItem == null)
  1068. return null;
  1069. agentItem.Folder = folderId;
  1070. AddInventoryItem(avatarId, agentItem);
  1071. return agentItem;
  1072. }
  1073. }
  1074. /// <summary>
  1075. /// Copy a task (prim) inventory item to another task (prim)
  1076. /// </summary>
  1077. /// <param name="destId"></param>
  1078. /// <param name="part"></param>
  1079. /// <param name="itemId"></param>
  1080. public void MoveTaskInventoryItem(UUID destId, SceneObjectPart part, UUID itemId)
  1081. {
  1082. TaskInventoryItem srcTaskItem = part.Inventory.GetInventoryItem(itemId);
  1083. if (srcTaskItem == null)
  1084. {
  1085. m_log.ErrorFormat(
  1086. "[PRIM INVENTORY]: Tried to retrieve item ID {0} from prim {1}, {2} for moving"
  1087. + " but the item does not exist in this inventory",
  1088. itemId, part.Name, part.UUID);
  1089. return;
  1090. }
  1091. SceneObjectPart destPart = GetSceneObjectPart(destId);
  1092. if (destPart == null)
  1093. {
  1094. m_log.ErrorFormat(
  1095. "[PRIM INVENTORY]: " +
  1096. "Could not find prim for ID {0}",
  1097. destId);
  1098. return;
  1099. }
  1100. // Can't transfer this
  1101. //
  1102. if ((part.OwnerID != destPart.OwnerID) && ((srcTaskItem.CurrentPermissions & (uint)PermissionMask.Transfer) == 0))
  1103. return;
  1104. if (part.OwnerID != destPart.OwnerID && (part.GetEffectiveObjectFlags() & (uint)PrimFlags.AllowInventoryDrop) == 0)
  1105. {
  1106. // object cannot copy items to an object owned by a different owner
  1107. // unless llAllowInventoryDrop has been called
  1108. return;
  1109. }
  1110. // must have both move and modify permission to put an item in an object
  1111. if ((part.OwnerMask & ((uint)PermissionMask.Move | (uint)PermissionMask.Modify)) == 0)
  1112. {
  1113. return;
  1114. }
  1115. TaskInventoryItem destTaskItem = new TaskInventoryItem();
  1116. destTaskItem.ItemID = UUID.Random();
  1117. destTaskItem.CreatorID = srcTaskItem.CreatorID;
  1118. destTaskItem.AssetID = srcTaskItem.AssetID;
  1119. destTaskItem.GroupID = destPart.GroupID;
  1120. destTaskItem.OwnerID = destPart.OwnerID;
  1121. destTaskItem.ParentID = destPart.UUID;
  1122. destTaskItem.ParentPartID = destPart.UUID;
  1123. destTaskItem.BasePermissions = srcTaskItem.BasePermissions;
  1124. destTaskItem.EveryonePermissions = srcTaskItem.EveryonePermissions;
  1125. destTaskItem.GroupPermissions = srcTaskItem.GroupPermissions;
  1126. destTaskItem.CurrentPermissions = srcTaskItem.CurrentPermissions;
  1127. destTaskItem.NextPermissions = srcTaskItem.NextPermissions;
  1128. destTaskItem.Flags = srcTaskItem.Flags;
  1129. if (destPart.OwnerID != part.OwnerID)
  1130. {
  1131. if (Permissions.PropagatePermissions())
  1132. {
  1133. destTaskItem.CurrentPermissions = srcTaskItem.CurrentPermissions &
  1134. srcTaskItem.NextPermissions;
  1135. destTaskItem.GroupPermissions = srcTaskItem.GroupPermissions &
  1136. srcTaskItem.NextPermissions;
  1137. destTaskItem.EveryonePermissions = srcTaskItem.EveryonePermissions &
  1138. srcTaskItem.NextPermissions;
  1139. destTaskItem.BasePermissions = srcTaskItem.BasePermissions &
  1140. srcTaskItem.NextPermissions;
  1141. destTaskItem.CurrentPermissions |= 8; // Slam!
  1142. }
  1143. }
  1144. destTaskItem.Description = srcTaskItem.Description;
  1145. destTaskItem.Name = srcTaskItem.Name;
  1146. destTaskItem.InvType = srcTaskItem.InvType;
  1147. destTaskItem.Type = srcTaskItem.Type;
  1148. destPart.Inventory.AddInventoryItem(destTaskItem, part.OwnerID != destPart.OwnerID);
  1149. if ((srcTaskItem.CurrentPermissions & (uint)PermissionMask.Copy) == 0)
  1150. part.Inventory.RemoveInventoryItem(itemId);
  1151. ScenePresence avatar;
  1152. if (TryGetAvatar(srcTaskItem.OwnerID, out avatar))
  1153. {
  1154. destPart.GetProperties(avatar.ControllingClient);
  1155. }
  1156. }
  1157. public UUID MoveTaskInventoryItems(UUID destID, string category, SceneObjectPart host, List<UUID> items)
  1158. {
  1159. CachedUserInfo profile = CommsManager.UserProfileCacheService.GetUserDetails(destID);
  1160. if (profile == null || profile.RootFolder == null)
  1161. {
  1162. m_log.ErrorFormat(
  1163. "[PRIM INVENTORY]: " +
  1164. "Avatar {0} cannot be found to add items",
  1165. destID);
  1166. return UUID.Zero;
  1167. }
  1168. UUID newFolderID = UUID.Random();
  1169. profile.CreateFolder(category, newFolderID, 0xffff, profile.RootFolder.ID);
  1170. foreach (UUID itemID in items)
  1171. {
  1172. InventoryItemBase agentItem = CreateAgentInventoryItemFromTask(destID, host, itemID);
  1173. if (agentItem != null)
  1174. {
  1175. agentItem.Folder = newFolderID;
  1176. AddInventoryItem(destID, agentItem);
  1177. }
  1178. }
  1179. ScenePresence avatar;
  1180. if (TryGetAvatar(destID, out avatar))
  1181. {
  1182. profile.SendInventoryDecendents(avatar.ControllingClient,
  1183. profile.RootFolder.ID, true, false);
  1184. profile.SendInventoryDecendents(avatar.ControllingClient,
  1185. newFolderID, false, true);
  1186. }
  1187. return newFolderID;
  1188. }
  1189. /// <summary>
  1190. /// Update an item in a prim (task) inventory.
  1191. /// This method does not handle scripts, <see>RezScript(IClientAPI, UUID, unit)</see>
  1192. /// </summary>
  1193. /// <param name="remoteClient"></param>
  1194. /// <param name="transactionID"></param>
  1195. /// <param name="itemInfo"></param>
  1196. /// <param name="primLocalID"></param>
  1197. public void UpdateTaskInventory(IClientAPI remoteClient, UUID transactionID, TaskInventoryItem itemInfo,
  1198. uint primLocalID)
  1199. {
  1200. UUID itemID = itemInfo.ItemID;
  1201. // Find the prim we're dealing with
  1202. SceneObjectPart part = GetSceneObjectPart(primLocalID);
  1203. if (part != null)
  1204. {
  1205. TaskInventoryItem currentItem = part.Inventory.GetInventoryItem(itemID);
  1206. bool allowInventoryDrop = (part.GetEffectiveObjectFlags()
  1207. & (uint)PrimFlags.AllowInventoryDrop) != 0;
  1208. // Explicity allow anyone to add to the inventory if the
  1209. // AllowInventoryDrop flag has been set. Don't however let
  1210. // them update an item unless they pass the external checks
  1211. //
  1212. if (!Permissions.CanEditObjectInventory(part.UUID, remoteClient.AgentId)
  1213. && (currentItem != null || !allowInventoryDrop))
  1214. return;
  1215. if (currentItem == null)
  1216. {
  1217. UUID copyID = UUID.Random();
  1218. if (itemID != UUID.Zero)
  1219. {
  1220. CachedUserInfo userInfo = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  1221. if (userInfo != null && userInfo.RootFolder != null)
  1222. {
  1223. InventoryItemBase item = userInfo.RootFolder.FindItem(itemID);
  1224. // Try library
  1225. // XXX clumsy, possibly should be one call
  1226. if (null == item)
  1227. {
  1228. item = CommsManager.UserProfileCacheService.LibraryRoot.FindItem(itemID);
  1229. }
  1230. if (item != null)
  1231. {
  1232. part.ParentGroup.AddInventoryItem(remoteClient, primLocalID, item, copyID);
  1233. m_log.InfoFormat(
  1234. "[PRIM INVENTORY]: Update with item {0} requested of prim {1} for {2}",
  1235. item.Name, primLocalID, remoteClient.Name);
  1236. part.GetProperties(remoteClient);
  1237. if (!Permissions.BypassPermissions())
  1238. {
  1239. if ((item.CurrentPermissions & (uint)PermissionMask.Copy) == 0)
  1240. RemoveInventoryItem(remoteClient, itemID);
  1241. }
  1242. }
  1243. else
  1244. {
  1245. m_log.ErrorFormat(
  1246. "[PRIM INVENTORY]: Could not find inventory item {0} to update for {1}!",
  1247. itemID, remoteClient.Name);
  1248. }
  1249. }
  1250. }
  1251. }
  1252. else // Updating existing item with new perms etc
  1253. {
  1254. IAgentAssetTransactions agentTransactions = this.RequestModuleInterface<IAgentAssetTransactions>();
  1255. if (agentTransactions != null)
  1256. {
  1257. agentTransactions.HandleTaskItemUpdateFromTransaction(
  1258. remoteClient, part, transactionID, currentItem);
  1259. }
  1260. if (part.Inventory.UpdateInventoryItem(itemInfo))
  1261. part.GetProperties(remoteClient);
  1262. }
  1263. }
  1264. else
  1265. {
  1266. m_log.WarnFormat(
  1267. "[PRIM INVENTORY]: " +
  1268. "Update with item {0} requested of prim {1} for {2} but this prim does not exist",
  1269. itemID, primLocalID, remoteClient.Name);
  1270. }
  1271. }
  1272. /// <summary>
  1273. /// Rez a script into a prim's inventory, either ex nihilo or from an existing avatar inventory
  1274. /// </summary>
  1275. /// <param name="remoteClient"></param>
  1276. /// <param name="itemID"> </param>
  1277. /// <param name="localID"></param>
  1278. public void RezScript(IClientAPI remoteClient, InventoryItemBase itemBase, UUID transactionID, uint localID)
  1279. {
  1280. UUID itemID = itemBase.ID;
  1281. UUID copyID = UUID.Random();
  1282. if (itemID != UUID.Zero) // transferred from an avatar inventory to the prim's inventory
  1283. {
  1284. CachedUserInfo userInfo = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  1285. if (userInfo != null && userInfo.RootFolder != null)
  1286. {
  1287. InventoryItemBase item = userInfo.RootFolder.FindItem(itemID);
  1288. // Try library
  1289. // XXX clumsy, possibly should be one call
  1290. if (null == item)
  1291. {
  1292. item = CommsManager.UserProfileCacheService.LibraryRoot.FindItem(itemID);
  1293. }
  1294. if (item != null)
  1295. {
  1296. SceneObjectPart part = GetSceneObjectPart(localID);
  1297. if (part != null)
  1298. {
  1299. if (!Permissions.CanEditObjectInventory(part.UUID, remoteClient.AgentId))
  1300. return;
  1301. part.ParentGroup.AddInventoryItem(remoteClient, localID, item, copyID);
  1302. // TODO: switch to posting on_rez here when scripts
  1303. // have state in inventory
  1304. part.Inventory.CreateScriptInstance(copyID, 0, false, DefaultScriptEngine, 0);
  1305. // m_log.InfoFormat("[PRIMINVENTORY]: " +
  1306. // "Rezzed script {0} into prim local ID {1} for user {2}",
  1307. // item.inventoryName, localID, remoteClient.Name);
  1308. part.GetProperties(remoteClient);
  1309. }
  1310. else
  1311. {
  1312. m_log.ErrorFormat(
  1313. "[PRIM INVENTORY]: " +
  1314. "Could not rez script {0} into prim local ID {1} for user {2}"
  1315. + " because the prim could not be found in the region!",
  1316. item.Name, localID, remoteClient.Name);
  1317. }
  1318. }
  1319. else
  1320. {
  1321. m_log.ErrorFormat(
  1322. "[PRIM INVENTORY]: Could not find script inventory item {0} to rez for {1}!",
  1323. itemID, remoteClient.Name);
  1324. }
  1325. }
  1326. }
  1327. else // script has been rezzed directly into a prim's inventory
  1328. {
  1329. SceneObjectPart part = GetSceneObjectPart(itemBase.Folder);
  1330. if (part == null)
  1331. return;
  1332. if (part.OwnerID != remoteClient.AgentId)
  1333. {
  1334. // Group permissions
  1335. if ((part.GroupID == UUID.Zero) || (remoteClient.GetGroupPowers(part.GroupID) == 0) || ((part.GroupMask & (uint)PermissionMask.Modify) == 0))
  1336. return;
  1337. } else {
  1338. if ((part.OwnerMask & (uint)PermissionMask.Modify) == 0)
  1339. return;
  1340. }
  1341. if (!Permissions.CanCreateObjectInventory(
  1342. itemBase.InvType, part.UUID, remoteClient.AgentId))
  1343. return;
  1344. AssetBase asset = CreateAsset(itemBase.Name, itemBase.Description, (sbyte)itemBase.AssetType, Encoding.ASCII.GetBytes("default\n{\n state_entry()\n {\n llSay(0, \"Script running\");\n }\n}"));
  1345. AssetService.Store(asset);
  1346. TaskInventoryItem taskItem = new TaskInventoryItem();
  1347. taskItem.ResetIDs(itemBase.Folder);
  1348. taskItem.ParentID = itemBase.Folder;
  1349. taskItem.CreationDate = (uint)itemBase.CreationDate;
  1350. taskItem.Name = itemBase.Name;
  1351. taskItem.Description = itemBase.Description;
  1352. taskItem.Type = itemBase.AssetType;
  1353. taskItem.InvType = itemBase.InvType;
  1354. taskItem.OwnerID = itemBase.Owner;
  1355. taskItem.CreatorID = itemBase.CreatorIdAsUuid;
  1356. taskItem.BasePermissions = itemBase.BasePermissions;
  1357. taskItem.CurrentPermissions = itemBase.CurrentPermissions;
  1358. taskItem.EveryonePermissions = itemBase.EveryOnePermissions;
  1359. taskItem.GroupPermissions = itemBase.GroupPermissions;
  1360. taskItem.NextPermissions = itemBase.NextPermissions;
  1361. taskItem.GroupID = itemBase.GroupID;
  1362. taskItem.GroupPermissions = 0;
  1363. taskItem.Flags = itemBase.Flags;
  1364. taskItem.PermsGranter = UUID.Zero;
  1365. taskItem.PermsMask = 0;
  1366. taskItem.AssetID = asset.FullID;
  1367. part.Inventory.AddInventoryItem(taskItem, false);
  1368. part.GetProperties(remoteClient);
  1369. part.Inventory.CreateScriptInstance(taskItem, 0, false, DefaultScriptEngine, 0);
  1370. }
  1371. }
  1372. /// <summary>
  1373. /// Rez a script into a prim's inventory from another prim
  1374. /// </summary>
  1375. /// <param name="remoteClient"></param>
  1376. /// <param name="itemID"> </param>
  1377. /// <param name="localID"></param>
  1378. public void RezScript(UUID srcId, SceneObjectPart srcPart, UUID destId, int pin, int running, int start_param)
  1379. {
  1380. TaskInventoryItem srcTaskItem = srcPart.Inventory.GetInventoryItem(srcId);
  1381. if (srcTaskItem == null)
  1382. {
  1383. m_log.ErrorFormat(
  1384. "[PRIM INVENTORY]: Tried to retrieve item ID {0} from prim {1}, {2} for rezzing a script but the "
  1385. + " item does not exist in this inventory",
  1386. srcId, srcPart.Name, srcPart.UUID);
  1387. return;
  1388. }
  1389. SceneObjectPart destPart = GetSceneObjectPart(destId);
  1390. if (destPart == null)
  1391. {
  1392. m_log.ErrorFormat(
  1393. "[PRIM INVENTORY]: " +
  1394. "Could not find script for ID {0}",
  1395. destId);
  1396. return;
  1397. }
  1398. // Must own the object, and have modify rights
  1399. if (srcPart.OwnerID != destPart.OwnerID)
  1400. {
  1401. // Group permissions
  1402. if ((destPart.GroupID == UUID.Zero) || (destPart.GroupID != srcPart.GroupID) ||
  1403. ((destPart.GroupMask & (uint)PermissionMask.Modify) == 0))
  1404. return;
  1405. } else {
  1406. if ((destPart.OwnerMask & (uint)PermissionMask.Modify) == 0)
  1407. return;
  1408. }
  1409. if (destPart.ScriptAccessPin != pin)
  1410. {
  1411. m_log.WarnFormat(
  1412. "[PRIM INVENTORY]: " +
  1413. "Script in object {0} : {1}, attempted to load script {2} : {3} into object {4} : {5} with invalid pin {6}",
  1414. srcPart.Name, srcId, srcTaskItem.Name, srcTaskItem.ItemID, destPart.Name, destId, pin);
  1415. // the LSL Wiki says we are supposed to shout on the DEBUG_CHANNEL -
  1416. // "Object: Task Object trying to illegally load script onto task Other_Object!"
  1417. // How do we shout from in here?
  1418. return;
  1419. }
  1420. TaskInventoryItem destTaskItem = new TaskInventoryItem();
  1421. destTaskItem.ItemID = UUID.Random();
  1422. destTaskItem.CreatorID = srcTaskItem.CreatorID;
  1423. destTaskItem.AssetID = srcTaskItem.AssetID;
  1424. destTaskItem.GroupID = destPart.GroupID;
  1425. destTaskItem.OwnerID = destPart.OwnerID;
  1426. destTaskItem.ParentID = destPart.UUID;
  1427. destTaskItem.ParentPartID = destPart.UUID;
  1428. destTaskItem.BasePermissions = srcTaskItem.BasePermissions;
  1429. destTaskItem.EveryonePermissions = srcTaskItem.EveryonePermissions;
  1430. destTaskItem.GroupPermissions = srcTaskItem.GroupPermissions;
  1431. destTaskItem.CurrentPermissions = srcTaskItem.CurrentPermissions;
  1432. destTaskItem.NextPermissions = srcTaskItem.NextPermissions;
  1433. destTaskItem.Flags = srcTaskItem.Flags;
  1434. if (destPart.OwnerID != srcPart.OwnerID)
  1435. {
  1436. if (Permissions.PropagatePermissions())
  1437. {
  1438. destTaskItem.CurrentPermissions = srcTaskItem.CurrentPermissions &
  1439. srcTaskItem.NextPermissions;
  1440. destTaskItem.GroupPermissions = srcTaskItem.GroupPermissions &
  1441. srcTaskItem.NextPermissions;
  1442. destTaskItem.EveryonePermissions = srcTaskItem.EveryonePermissions &
  1443. srcTaskItem.NextPermissions;
  1444. destTaskItem.BasePermissions = srcTaskItem.BasePermissions &
  1445. srcTaskItem.NextPermissions;
  1446. destTaskItem.CurrentPermissions |= 8; // Slam!
  1447. }
  1448. }
  1449. destTaskItem.Description = srcTaskItem.Description;
  1450. destTaskItem.Name = srcTaskItem.Name;
  1451. destTaskItem.InvType = srcTaskItem.InvType;
  1452. destTaskItem.Type = srcTaskItem.Type;
  1453. destPart.Inventory.AddInventoryItemExclusive(destTaskItem, false);
  1454. if (running > 0)
  1455. {
  1456. destPart.Inventory.CreateScriptInstance(destTaskItem, start_param, false, DefaultScriptEngine, 0);
  1457. }
  1458. ScenePresence avatar;
  1459. if (TryGetAvatar(srcTaskItem.OwnerID, out avatar))
  1460. {
  1461. destPart.GetProperties(avatar.ControllingClient);
  1462. }
  1463. }
  1464. /// <summary>
  1465. /// Called when one or more objects are removed from the environment into inventory.
  1466. /// </summary>
  1467. /// <param name="remoteClient"></param>
  1468. /// <param name="localID"></param>
  1469. /// <param name="groupID"></param>
  1470. /// <param name="action"></param>
  1471. /// <param name="destinationID"></param>
  1472. public virtual void DeRezObject(IClientAPI remoteClient, List<uint> localIDs,
  1473. UUID groupID, DeRezAction action, UUID destinationID)
  1474. {
  1475. foreach (uint localID in localIDs)
  1476. {
  1477. DeRezObject(remoteClient, localID, groupID, action, destinationID);
  1478. }
  1479. }
  1480. /// <summary>
  1481. /// Called when an object is removed from the environment into inventory.
  1482. /// </summary>
  1483. /// <param name="remoteClient"></param>
  1484. /// <param name="localID"></param>
  1485. /// <param name="groupID"></param>
  1486. /// <param name="action"></param>
  1487. /// <param name="destinationID"></param>
  1488. public virtual void DeRezObject(IClientAPI remoteClient, uint localID,
  1489. UUID groupID, DeRezAction action, UUID destinationID)
  1490. {
  1491. SceneObjectPart part = GetSceneObjectPart(localID);
  1492. if (part == null)
  1493. return;
  1494. if (part.ParentGroup == null || part.ParentGroup.IsDeleted)
  1495. return;
  1496. // Can't delete child prims
  1497. if (part != part.ParentGroup.RootPart)
  1498. return;
  1499. SceneObjectGroup grp = part.ParentGroup;
  1500. //force a database backup/update on this SceneObjectGroup
  1501. //So that we know the database is upto date, for when deleting the object from it
  1502. ForceSceneObjectBackup(grp);
  1503. bool permissionToTake = false;
  1504. bool permissionToDelete = false;
  1505. if (action == DeRezAction.SaveToExistingUserInventoryItem)
  1506. {
  1507. if (grp.OwnerID == remoteClient.AgentId && grp.RootPart.FromUserInventoryItemID != UUID.Zero)
  1508. {
  1509. permissionToTake = true;
  1510. permissionToDelete = false;
  1511. }
  1512. }
  1513. else if (action == DeRezAction.TakeCopy)
  1514. {
  1515. permissionToTake =
  1516. Permissions.CanTakeCopyObject(
  1517. grp.UUID,
  1518. remoteClient.AgentId);
  1519. }
  1520. else if (action == DeRezAction.GodTakeCopy)
  1521. {
  1522. permissionToTake =
  1523. Permissions.IsGod(
  1524. remoteClient.AgentId);
  1525. }
  1526. else if (action == DeRezAction.Take)
  1527. {
  1528. permissionToTake =
  1529. Permissions.CanTakeObject(
  1530. grp.UUID,
  1531. remoteClient.AgentId);
  1532. //If they can take, they can delete!
  1533. permissionToDelete = permissionToTake;
  1534. }
  1535. else if (action == DeRezAction.Delete)
  1536. {
  1537. permissionToTake =
  1538. Permissions.CanDeleteObject(
  1539. grp.UUID,
  1540. remoteClient.AgentId);
  1541. permissionToDelete = permissionToTake;
  1542. }
  1543. else if (action == DeRezAction.Return)
  1544. {
  1545. if (remoteClient != null)
  1546. {
  1547. permissionToTake =
  1548. Permissions.CanReturnObject(
  1549. grp.UUID,
  1550. remoteClient.AgentId);
  1551. permissionToDelete = permissionToTake;
  1552. if (permissionToDelete)
  1553. {
  1554. AddReturn(grp.OwnerID, grp.Name, grp.AbsolutePosition, "parcel owner return");
  1555. }
  1556. }
  1557. else // Auto return passes through here with null agent
  1558. {
  1559. permissionToTake = true;
  1560. permissionToDelete = true;
  1561. }
  1562. }
  1563. else
  1564. {
  1565. m_log.DebugFormat(
  1566. "[AGENT INVENTORY]: Ignoring unexpected derez action {0} for {1}", action, remoteClient.Name);
  1567. return;
  1568. }
  1569. if (permissionToTake)
  1570. {
  1571. m_asyncSceneObjectDeleter.DeleteToInventory(
  1572. action, destinationID, grp, remoteClient,
  1573. permissionToDelete);
  1574. }
  1575. else if (permissionToDelete)
  1576. {
  1577. DeleteSceneObject(grp, false);
  1578. }
  1579. }
  1580. private bool WaitForInventory(CachedUserInfo info)
  1581. {
  1582. // 200 Seconds wait. This is called in the context of the
  1583. // background delete thread, so we can afford to waste time
  1584. // here.
  1585. //
  1586. int count = 200;
  1587. while (count > 0)
  1588. {
  1589. System.Threading.Thread.Sleep(100);
  1590. count--;
  1591. if (info.HasReceivedInventory)
  1592. return true;
  1593. }
  1594. m_log.DebugFormat("Timed out waiting for inventory of user {0}",
  1595. info.UserProfile.ID.ToString());
  1596. return false;
  1597. }
  1598. /// <summary>
  1599. /// Delete a scene object from a scene and place in the given avatar's inventory.
  1600. /// Returns the UUID of the newly created asset.
  1601. /// </summary>
  1602. /// <param name="action"></param>
  1603. /// <param name="folderID"></param>
  1604. /// <param name="objectGroup"></param>
  1605. /// <param name="remoteClient"> </param>
  1606. public virtual UUID DeleteToInventory(DeRezAction action, UUID folderID,
  1607. SceneObjectGroup objectGroup, IClientAPI remoteClient)
  1608. {
  1609. UUID assetID = UUID.Zero;
  1610. string sceneObjectXml = SceneObjectSerializer.ToOriginalXmlFormat(objectGroup);
  1611. // Get the user info of the item destination
  1612. //
  1613. CachedUserInfo userInfo;
  1614. if (action == DeRezAction.Take || action == DeRezAction.TakeCopy ||
  1615. action == DeRezAction.SaveToExistingUserInventoryItem)
  1616. {
  1617. // Take or take copy require a taker
  1618. // Saving changes requires a local user
  1619. //
  1620. if (remoteClient == null)
  1621. return UUID.Zero;
  1622. userInfo = CommsManager.UserProfileCacheService.GetUserDetails(
  1623. remoteClient.AgentId);
  1624. }
  1625. else
  1626. {
  1627. // All returns / deletes go to the object owner
  1628. //
  1629. userInfo = CommsManager.UserProfileCacheService.GetUserDetails(
  1630. objectGroup.RootPart.OwnerID);
  1631. }
  1632. if (userInfo == null) // Can't proceed
  1633. {
  1634. return UUID.Zero;
  1635. }
  1636. if (!userInfo.HasReceivedInventory)
  1637. {
  1638. // Async inventory requests will queue, but they will never
  1639. // execute unless inventory is actually fetched
  1640. //
  1641. userInfo.FetchInventory();
  1642. }
  1643. // If we're returning someone's item, it goes back to the
  1644. // owner's Lost And Found folder.
  1645. // Delete is treated like return in this case
  1646. // Deleting your own items makes them go to trash
  1647. //
  1648. InventoryFolderBase folder = null;
  1649. InventoryItemBase item = null;
  1650. if (DeRezAction.SaveToExistingUserInventoryItem == action)
  1651. {
  1652. item = userInfo.RootFolder.FindItem(
  1653. objectGroup.RootPart.FromUserInventoryItemID);
  1654. if (null == item)
  1655. {
  1656. m_log.DebugFormat(
  1657. "[AGENT INVENTORY]: Object {0} {1} scheduled for save to inventory has already been deleted.",
  1658. objectGroup.Name, objectGroup.UUID);
  1659. return UUID.Zero;
  1660. }
  1661. }
  1662. else
  1663. {
  1664. // Folder magic
  1665. //
  1666. if (action == DeRezAction.Delete)
  1667. {
  1668. // Deleting someone else's item
  1669. //
  1670. if (remoteClient == null ||
  1671. objectGroup.OwnerID != remoteClient.AgentId)
  1672. {
  1673. // Folder skeleton may not be loaded and we
  1674. // have to wait for the inventory to find
  1675. // the destination folder
  1676. //
  1677. if (!WaitForInventory(userInfo))
  1678. return UUID.Zero;
  1679. folder = userInfo.FindFolderForType(
  1680. (int)AssetType.LostAndFoundFolder);
  1681. }
  1682. else
  1683. {
  1684. // Assume inventory skeleton was loaded during login
  1685. // and all folders can be found
  1686. //
  1687. folder = userInfo.FindFolderForType(
  1688. (int)AssetType.TrashFolder);
  1689. }
  1690. }
  1691. else if (action == DeRezAction.Return)
  1692. {
  1693. // Wait if needed
  1694. //
  1695. if (!userInfo.HasReceivedInventory)
  1696. {
  1697. if (!WaitForInventory(userInfo))
  1698. return UUID.Zero;
  1699. }
  1700. // Dump to lost + found unconditionally
  1701. //
  1702. folder = userInfo.FindFolderForType(
  1703. (int)AssetType.LostAndFoundFolder);
  1704. }
  1705. if (folderID == UUID.Zero && folder == null)
  1706. {
  1707. // Catch all. Use lost & found
  1708. //
  1709. if (!userInfo.HasReceivedInventory)
  1710. {
  1711. if (!WaitForInventory(userInfo))
  1712. return UUID.Zero;
  1713. }
  1714. folder = userInfo.FindFolderForType(
  1715. (int)AssetType.LostAndFoundFolder);
  1716. }
  1717. if (folder == null) // None of the above
  1718. {
  1719. folder = userInfo.RootFolder.FindFolder(folderID);
  1720. if (folder == null) // Nowhere to put it
  1721. {
  1722. return UUID.Zero;
  1723. }
  1724. }
  1725. item = new InventoryItemBase();
  1726. item.CreatorId = objectGroup.RootPart.CreatorID.ToString();
  1727. item.ID = UUID.Random();
  1728. item.InvType = (int)InventoryType.Object;
  1729. item.Folder = folder.ID;
  1730. item.Owner = userInfo.UserProfile.ID;
  1731. }
  1732. AssetBase asset = CreateAsset(
  1733. objectGroup.GetPartName(objectGroup.RootPart.LocalId),
  1734. objectGroup.GetPartDescription(objectGroup.RootPart.LocalId),
  1735. (sbyte)AssetType.Object,
  1736. Utils.StringToBytes(sceneObjectXml));
  1737. AssetService.Store(asset);
  1738. assetID = asset.FullID;
  1739. if (DeRezAction.SaveToExistingUserInventoryItem == action)
  1740. {
  1741. item.AssetID = asset.FullID;
  1742. userInfo.UpdateItem(item);
  1743. }
  1744. else
  1745. {
  1746. item.AssetID = asset.FullID;
  1747. if (remoteClient != null && (remoteClient.AgentId != objectGroup.RootPart.OwnerID) && Permissions.PropagatePermissions())
  1748. {
  1749. uint perms=objectGroup.GetEffectivePermissions();
  1750. uint nextPerms=(perms & 7) << 13;
  1751. if ((nextPerms & (uint)PermissionMask.Copy) == 0)
  1752. perms &= ~(uint)PermissionMask.Copy;
  1753. if ((nextPerms & (uint)PermissionMask.Transfer) == 0)
  1754. perms &= ~(uint)PermissionMask.Transfer;
  1755. if ((nextPerms & (uint)PermissionMask.Modify) == 0)
  1756. perms &= ~(uint)PermissionMask.Modify;
  1757. item.BasePermissions = perms & objectGroup.RootPart.NextOwnerMask;
  1758. item.CurrentPermissions = item.BasePermissions;
  1759. item.NextPermissions = objectGroup.RootPart.NextOwnerMask;
  1760. item.EveryOnePermissions = objectGroup.RootPart.EveryoneMask & objectGroup.RootPart.NextOwnerMask;
  1761. item.GroupPermissions = objectGroup.RootPart.GroupMask & objectGroup.RootPart.NextOwnerMask;
  1762. item.CurrentPermissions |= 8; // Slam!
  1763. }
  1764. else
  1765. {
  1766. item.BasePermissions = objectGroup.GetEffectivePermissions();
  1767. item.CurrentPermissions = objectGroup.GetEffectivePermissions();
  1768. item.NextPermissions = objectGroup.RootPart.NextOwnerMask;
  1769. item.EveryOnePermissions = objectGroup.RootPart.EveryoneMask;
  1770. item.GroupPermissions = objectGroup.RootPart.GroupMask;
  1771. item.CurrentPermissions |= 8; // Slam!
  1772. }
  1773. // TODO: add the new fields (Flags, Sale info, etc)
  1774. item.CreationDate = Util.UnixTimeSinceEpoch();
  1775. item.Description = asset.Description;
  1776. item.Name = asset.Name;
  1777. item.AssetType = asset.Type;
  1778. userInfo.AddItem(item);
  1779. if (remoteClient != null && item.Owner == remoteClient.AgentId)
  1780. {
  1781. remoteClient.SendInventoryItemCreateUpdate(item, 0);
  1782. }
  1783. else
  1784. {
  1785. ScenePresence notifyUser = GetScenePresence(item.Owner);
  1786. if (notifyUser != null)
  1787. {
  1788. notifyUser.ControllingClient.SendInventoryItemCreateUpdate(item, 0);
  1789. }
  1790. }
  1791. }
  1792. return assetID;
  1793. }
  1794. public void updateKnownAsset(IClientAPI remoteClient, SceneObjectGroup grp, UUID assetID, UUID agentID)
  1795. {
  1796. SceneObjectGroup objectGroup = grp;
  1797. if (objectGroup != null)
  1798. {
  1799. if (!grp.HasGroupChanged)
  1800. {
  1801. m_log.InfoFormat("[ATTACHMENT]: Save request for {0} which is unchanged", grp.UUID);
  1802. return;
  1803. }
  1804. m_log.InfoFormat(
  1805. "[ATTACHMENT]: Updating asset for attachment {0}, attachpoint {1}",
  1806. grp.UUID, grp.GetAttachmentPoint());
  1807. string sceneObjectXml = SceneObjectSerializer.ToOriginalXmlFormat(objectGroup);
  1808. CachedUserInfo userInfo =
  1809. CommsManager.UserProfileCacheService.GetUserDetails(agentID);
  1810. if (userInfo != null && userInfo.RootFolder != null)
  1811. {
  1812. Queue<InventoryFolderImpl> searchfolders = new Queue<InventoryFolderImpl>();
  1813. searchfolders.Enqueue(userInfo.RootFolder);
  1814. UUID foundFolder = UUID.Zero;
  1815. InventoryItemBase item = null;
  1816. // search through folders to find the asset.
  1817. while (searchfolders.Count > 0)
  1818. {
  1819. InventoryFolderImpl fld = searchfolders.Dequeue();
  1820. lock (fld)
  1821. {
  1822. if (fld != null)
  1823. {
  1824. if (fld.Items.ContainsKey(assetID))
  1825. {
  1826. item = fld.Items[assetID];
  1827. foundFolder = fld.ID;
  1828. searchfolders.Clear();
  1829. break;
  1830. }
  1831. else
  1832. {
  1833. foreach (InventoryFolderImpl subfld in fld.RequestListOfFolderImpls())
  1834. {
  1835. searchfolders.Enqueue(subfld);
  1836. }
  1837. }
  1838. }
  1839. }
  1840. }
  1841. if (foundFolder != UUID.Zero && item != null)
  1842. {
  1843. AssetBase asset = CreateAsset(
  1844. objectGroup.GetPartName(objectGroup.LocalId),
  1845. objectGroup.GetPartDescription(objectGroup.LocalId),
  1846. (sbyte)AssetType.Object,
  1847. Utils.StringToBytes(sceneObjectXml));
  1848. AssetService.Store(asset);
  1849. item.AssetID = asset.FullID;
  1850. item.Description = asset.Description;
  1851. item.Name = asset.Name;
  1852. item.AssetType = asset.Type;
  1853. item.InvType = (int)InventoryType.Object;
  1854. item.Folder = foundFolder;
  1855. userInfo.UpdateItem(item);
  1856. // this gets called when the agent loggs off!
  1857. if (remoteClient != null)
  1858. {
  1859. remoteClient.SendInventoryItemCreateUpdate(item, 0);
  1860. }
  1861. }
  1862. }
  1863. }
  1864. }
  1865. public UUID attachObjectAssetStore(IClientAPI remoteClient, SceneObjectGroup grp, UUID AgentId, out UUID itemID)
  1866. {
  1867. itemID = UUID.Zero;
  1868. if (grp != null)
  1869. {
  1870. string sceneObjectXml = SceneObjectSerializer.ToOriginalXmlFormat(grp);
  1871. CachedUserInfo userInfo =
  1872. CommsManager.UserProfileCacheService.GetUserDetails(AgentId);
  1873. if (userInfo != null)
  1874. {
  1875. AssetBase asset = CreateAsset(
  1876. grp.GetPartName(grp.LocalId),
  1877. grp.GetPartDescription(grp.LocalId),
  1878. (sbyte)AssetType.Object,
  1879. Utils.StringToBytes(sceneObjectXml));
  1880. AssetService.Store(asset);
  1881. InventoryItemBase item = new InventoryItemBase();
  1882. item.CreatorId = grp.RootPart.CreatorID.ToString();
  1883. item.Owner = remoteClient.AgentId;
  1884. item.ID = UUID.Random();
  1885. item.AssetID = asset.FullID;
  1886. item.Description = asset.Description;
  1887. item.Name = asset.Name;
  1888. item.AssetType = asset.Type;
  1889. item.InvType = (int)InventoryType.Object;
  1890. item.Folder = UUID.Zero; // Objects folder!
  1891. if ((remoteClient.AgentId != grp.RootPart.OwnerID) && Permissions.PropagatePermissions())
  1892. {
  1893. item.BasePermissions = grp.RootPart.NextOwnerMask;
  1894. item.CurrentPermissions = grp.RootPart.NextOwnerMask;
  1895. item.NextPermissions = grp.RootPart.NextOwnerMask;
  1896. item.EveryOnePermissions = grp.RootPart.EveryoneMask & grp.RootPart.NextOwnerMask;
  1897. item.GroupPermissions = grp.RootPart.GroupMask & grp.RootPart.NextOwnerMask;
  1898. }
  1899. else
  1900. {
  1901. item.BasePermissions = grp.RootPart.BaseMask;
  1902. item.CurrentPermissions = grp.RootPart.OwnerMask;
  1903. item.NextPermissions = grp.RootPart.NextOwnerMask;
  1904. item.EveryOnePermissions = grp.RootPart.EveryoneMask;
  1905. item.GroupPermissions = grp.RootPart.GroupMask;
  1906. }
  1907. item.CreationDate = Util.UnixTimeSinceEpoch();
  1908. // sets assetID so client can show asset as 'attached' in inventory
  1909. grp.SetFromAssetID(item.ID);
  1910. userInfo.AddItem(item);
  1911. remoteClient.SendInventoryItemCreateUpdate(item, 0);
  1912. itemID = item.ID;
  1913. return item.AssetID;
  1914. }
  1915. return UUID.Zero;
  1916. }
  1917. return UUID.Zero;
  1918. }
  1919. /// <summary>
  1920. /// Event Handler Rez an object into a scene
  1921. /// Calls the non-void event handler
  1922. /// </summary>
  1923. /// <param name="remoteClient"></param>
  1924. /// <param name="itemID"></param>
  1925. /// <param name="RayEnd"></param>
  1926. /// <param name="RayStart"></param>
  1927. /// <param name="RayTargetID"></param>
  1928. /// <param name="BypassRayCast"></param>
  1929. /// <param name="RayEndIsIntersection"></param>
  1930. /// <param name="EveryoneMask"></param>
  1931. /// <param name="GroupMask"></param>
  1932. /// <param name="RezSelected"></param>
  1933. /// <param name="RemoveItem"></param>
  1934. /// <param name="fromTaskID"></param>
  1935. public virtual void RezObject(IClientAPI remoteClient, UUID itemID, Vector3 RayEnd, Vector3 RayStart,
  1936. UUID RayTargetID, byte BypassRayCast, bool RayEndIsIntersection,
  1937. bool RezSelected, bool RemoveItem, UUID fromTaskID)
  1938. {
  1939. RezObject(
  1940. remoteClient, itemID, RayEnd, RayStart, RayTargetID, BypassRayCast, RayEndIsIntersection,
  1941. RezSelected, RemoveItem, fromTaskID, false);
  1942. }
  1943. /// <summary>
  1944. /// Rez an object into the scene from the user's inventory
  1945. /// </summary>
  1946. /// <param name="remoteClient"></param>
  1947. /// <param name="itemID"></param>
  1948. /// <param name="RayEnd"></param>
  1949. /// <param name="RayStart"></param>
  1950. /// <param name="RayTargetID"></param>
  1951. /// <param name="BypassRayCast"></param>
  1952. /// <param name="RayEndIsIntersection"></param>
  1953. /// <param name="RezSelected"></param>
  1954. /// <param name="RemoveItem"></param>
  1955. /// <param name="fromTaskID"></param>
  1956. /// <param name="attachment"></param>
  1957. /// <returns>The SceneObjectGroup rezzed or null if rez was unsuccessful.</returns>
  1958. public virtual SceneObjectGroup RezObject(IClientAPI remoteClient, UUID itemID, Vector3 RayEnd, Vector3 RayStart,
  1959. UUID RayTargetID, byte BypassRayCast, bool RayEndIsIntersection,
  1960. bool RezSelected, bool RemoveItem, UUID fromTaskID, bool attachment)
  1961. {
  1962. // Work out position details
  1963. byte bRayEndIsIntersection = (byte)0;
  1964. if (RayEndIsIntersection)
  1965. {
  1966. bRayEndIsIntersection = (byte)1;
  1967. }
  1968. else
  1969. {
  1970. bRayEndIsIntersection = (byte)0;
  1971. }
  1972. Vector3 scale = new Vector3(0.5f, 0.5f, 0.5f);
  1973. Vector3 pos = GetNewRezLocation(
  1974. RayStart, RayEnd, RayTargetID, Quaternion.Identity,
  1975. BypassRayCast, bRayEndIsIntersection,true,scale, false);
  1976. // Rez object
  1977. CachedUserInfo userInfo = CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  1978. if (userInfo != null)
  1979. {
  1980. // Do NOT use HasReceivedInventory here, this is called
  1981. // from within ItemReceive during login for attachments.
  1982. // Using HasReceivedInventory here will break attachment
  1983. // persistence!
  1984. //
  1985. if (userInfo.RootFolder != null)
  1986. {
  1987. InventoryItemBase item = userInfo.RootFolder.FindItem(itemID);
  1988. if (item != null)
  1989. {
  1990. AssetBase rezAsset = AssetService.Get(item.AssetID.ToString());
  1991. if (rezAsset != null)
  1992. {
  1993. UUID itemId = UUID.Zero;
  1994. // If we have permission to copy then link the rezzed object back to the user inventory
  1995. // item that it came from. This allows us to enable 'save object to inventory'
  1996. if (!Permissions.BypassPermissions())
  1997. {
  1998. if ((item.CurrentPermissions & (uint)PermissionMask.Copy) == (uint)PermissionMask.Copy)
  1999. {
  2000. itemId = item.ID;
  2001. }
  2002. }
  2003. else
  2004. {
  2005. // Brave new fullperm world
  2006. //
  2007. itemId = item.ID;
  2008. }
  2009. string xmlData = Utils.BytesToString(rezAsset.Data);
  2010. SceneObjectGroup group
  2011. = SceneObjectSerializer.FromOriginalXmlFormat(itemId, xmlData);
  2012. if (!Permissions.CanRezObject(
  2013. group.Children.Count, remoteClient.AgentId, pos)
  2014. && !attachment)
  2015. {
  2016. return null;
  2017. }
  2018. group.ResetIDs();
  2019. if (attachment)
  2020. group.RootPart.ObjectFlags |= (uint)PrimFlags.Phantom;
  2021. AddNewSceneObject(group, true);
  2022. // m_log.InfoFormat("ray end point for inventory rezz is {0} {1} {2} ", RayEnd.X, RayEnd.Y, RayEnd.Z);
  2023. // if attachment we set it's asset id so object updates can reflect that
  2024. // if not, we set it's position in world.
  2025. if (!attachment)
  2026. {
  2027. float offsetHeight = 0;
  2028. pos = GetNewRezLocation(
  2029. RayStart, RayEnd, RayTargetID, Quaternion.Identity,
  2030. BypassRayCast, bRayEndIsIntersection, true, group.GetAxisAlignedBoundingBox(out offsetHeight), false);
  2031. pos.Z += offsetHeight;
  2032. group.AbsolutePosition = pos;
  2033. // m_log.InfoFormat("rezx point for inventory rezz is {0} {1} {2} and offsetheight was {3}", pos.X, pos.Y, pos.Z, offsetHeight);
  2034. }
  2035. else
  2036. {
  2037. group.SetFromAssetID(itemID);
  2038. }
  2039. SceneObjectPart rootPart = null;
  2040. try
  2041. {
  2042. rootPart = group.GetChildPart(group.UUID);
  2043. }
  2044. catch (NullReferenceException)
  2045. {
  2046. string isAttachment = "";
  2047. if (attachment)
  2048. isAttachment = " Object was an attachment";
  2049. m_log.Error("[AGENT INVENTORY]: Error rezzing ItemID: " + itemID + " object has no rootpart." + isAttachment);
  2050. }
  2051. // Since renaming the item in the inventory does not affect the name stored
  2052. // in the serialization, transfer the correct name from the inventory to the
  2053. // object itself before we rez.
  2054. rootPart.Name = item.Name;
  2055. rootPart.Description = item.Description;
  2056. List<SceneObjectPart> partList = new List<SceneObjectPart>(group.Children.Values);
  2057. group.SetGroup(remoteClient.ActiveGroupId, remoteClient);
  2058. if (rootPart.OwnerID != item.Owner)
  2059. {
  2060. //Need to kill the for sale here
  2061. rootPart.ObjectSaleType = 0;
  2062. rootPart.SalePrice = 10;
  2063. if (Permissions.PropagatePermissions())
  2064. {
  2065. if ((item.CurrentPermissions & 8) != 0)
  2066. {
  2067. foreach (SceneObjectPart part in partList)
  2068. {
  2069. part.EveryoneMask = item.EveryOnePermissions;
  2070. part.NextOwnerMask = item.NextPermissions;
  2071. part.GroupMask = 0; // DO NOT propagate here
  2072. }
  2073. }
  2074. group.ApplyNextOwnerPermissions();
  2075. }
  2076. }
  2077. foreach (SceneObjectPart part in partList)
  2078. {
  2079. if (part.OwnerID != item.Owner)
  2080. {
  2081. part.LastOwnerID = part.OwnerID;
  2082. part.OwnerID = item.Owner;
  2083. part.Inventory.ChangeInventoryOwner(item.Owner);
  2084. }
  2085. else if (((item.CurrentPermissions & 8) != 0) && (!attachment)) // Slam!
  2086. {
  2087. part.EveryoneMask = item.EveryOnePermissions;
  2088. part.NextOwnerMask = item.NextPermissions;
  2089. part.GroupMask = 0; // DO NOT propagate here
  2090. }
  2091. }
  2092. rootPart.TrimPermissions();
  2093. if (!attachment)
  2094. {
  2095. if (group.RootPart.Shape.PCode == (byte)PCode.Prim)
  2096. {
  2097. group.ClearPartAttachmentData();
  2098. }
  2099. }
  2100. if (!attachment)
  2101. {
  2102. // Fire on_rez
  2103. group.CreateScriptInstances(0, true, DefaultScriptEngine, 0);
  2104. rootPart.ScheduleFullUpdate();
  2105. }
  2106. if (!Permissions.BypassPermissions())
  2107. {
  2108. if ((item.CurrentPermissions & (uint)PermissionMask.Copy) == 0)
  2109. {
  2110. // If this is done on attachments, no
  2111. // copy ones will be lost, so avoid it
  2112. //
  2113. if (!attachment)
  2114. userInfo.DeleteItem(item.ID);
  2115. }
  2116. }
  2117. return rootPart.ParentGroup;
  2118. }
  2119. }
  2120. }
  2121. else
  2122. m_log.WarnFormat("[AGENT INVENTORY]: Root folder not found in {0}", RegionInfo.RegionName);
  2123. }
  2124. else
  2125. m_log.WarnFormat("[AGENT INVENTORY]: User profile not found in {0}", RegionInfo.RegionName);
  2126. return null;
  2127. }
  2128. /// <summary>
  2129. /// Rez an object into the scene from a prim's inventory.
  2130. /// </summary>
  2131. /// <param name="sourcePart"></param>
  2132. /// <param name="item"></param>
  2133. /// <param name="pos"></param>
  2134. /// <param name="rot"></param>
  2135. /// <param name="vel"></param>
  2136. /// <param name="param"></param>
  2137. /// <returns>The SceneObjectGroup rezzed or null if rez was unsuccessful</returns>
  2138. public virtual SceneObjectGroup RezObject(
  2139. SceneObjectPart sourcePart, TaskInventoryItem item,
  2140. Vector3 pos, Quaternion rot, Vector3 vel, int param)
  2141. {
  2142. // Rez object
  2143. if (item != null)
  2144. {
  2145. UUID ownerID = item.OwnerID;
  2146. AssetBase rezAsset = AssetService.Get(item.AssetID.ToString());
  2147. if (rezAsset != null)
  2148. {
  2149. string xmlData = Utils.BytesToString(rezAsset.Data);
  2150. SceneObjectGroup group = SceneObjectSerializer.FromOriginalXmlFormat(xmlData);
  2151. if (!Permissions.CanRezObject(group.Children.Count, ownerID, pos))
  2152. {
  2153. return null;
  2154. }
  2155. group.ResetIDs();
  2156. AddNewSceneObject(group, true);
  2157. // we set it's position in world.
  2158. group.AbsolutePosition = pos;
  2159. SceneObjectPart rootPart = group.GetChildPart(group.UUID);
  2160. // Since renaming the item in the inventory does not affect the name stored
  2161. // in the serialization, transfer the correct name from the inventory to the
  2162. // object itself before we rez.
  2163. rootPart.Name = item.Name;
  2164. rootPart.Description = item.Description;
  2165. List<SceneObjectPart> partList = new List<SceneObjectPart>(group.Children.Values);
  2166. group.SetGroup(sourcePart.GroupID, null);
  2167. if (rootPart.OwnerID != item.OwnerID)
  2168. {
  2169. if (Permissions.PropagatePermissions())
  2170. {
  2171. if ((item.CurrentPermissions & 8) != 0)
  2172. {
  2173. foreach (SceneObjectPart part in partList)
  2174. {
  2175. part.EveryoneMask = item.EveryonePermissions;
  2176. part.NextOwnerMask = item.NextPermissions;
  2177. }
  2178. }
  2179. group.ApplyNextOwnerPermissions();
  2180. }
  2181. }
  2182. foreach (SceneObjectPart part in partList)
  2183. {
  2184. if (part.OwnerID != item.OwnerID)
  2185. {
  2186. part.LastOwnerID = part.OwnerID;
  2187. part.OwnerID = item.OwnerID;
  2188. part.Inventory.ChangeInventoryOwner(item.OwnerID);
  2189. }
  2190. else if ((item.CurrentPermissions & 8) != 0) // Slam!
  2191. {
  2192. part.EveryoneMask = item.EveryonePermissions;
  2193. part.NextOwnerMask = item.NextPermissions;
  2194. }
  2195. }
  2196. rootPart.TrimPermissions();
  2197. if (group.RootPart.Shape.PCode == (byte)PCode.Prim)
  2198. {
  2199. group.ClearPartAttachmentData();
  2200. }
  2201. group.UpdateGroupRotation(rot);
  2202. //group.ApplyPhysics(m_physicalPrim);
  2203. if (group.RootPart.PhysActor != null && group.RootPart.PhysActor.IsPhysical && vel != Vector3.Zero)
  2204. {
  2205. group.RootPart.ApplyImpulse((vel * group.GetMass()), false);
  2206. group.Velocity = vel;
  2207. rootPart.ScheduleFullUpdate();
  2208. }
  2209. group.CreateScriptInstances(param, true, DefaultScriptEngine, 2);
  2210. rootPart.ScheduleFullUpdate();
  2211. if (!Permissions.BypassPermissions())
  2212. {
  2213. if ((item.CurrentPermissions & (uint)PermissionMask.Copy) == 0)
  2214. sourcePart.Inventory.RemoveInventoryItem(item.ItemID);
  2215. }
  2216. return rootPart.ParentGroup;
  2217. }
  2218. }
  2219. return null;
  2220. }
  2221. public virtual bool returnObjects(SceneObjectGroup[] returnobjects, UUID AgentId)
  2222. {
  2223. foreach (SceneObjectGroup grp in returnobjects)
  2224. {
  2225. AddReturn(grp.OwnerID, grp.Name, grp.AbsolutePosition, "parcel owner return");
  2226. DeRezObject(null, grp.RootPart.LocalId,
  2227. grp.RootPart.GroupID, DeRezAction.Return, UUID.Zero);
  2228. }
  2229. return true;
  2230. }
  2231. public void SetScriptRunning(IClientAPI controllingClient, UUID objectID, UUID itemID, bool running)
  2232. {
  2233. SceneObjectPart part = GetSceneObjectPart(objectID);
  2234. if (part == null)
  2235. return;
  2236. if (running)
  2237. EventManager.TriggerStartScript(part.LocalId, itemID);
  2238. else
  2239. EventManager.TriggerStopScript(part.LocalId, itemID);
  2240. }
  2241. public UUID RezSingleAttachment(IClientAPI remoteClient, UUID itemID,
  2242. uint AttachmentPt)
  2243. {
  2244. SceneObjectGroup att = m_sceneGraph.RezSingleAttachment(remoteClient, itemID, AttachmentPt);
  2245. if (att == null)
  2246. {
  2247. DetachSingleAttachmentToInv(itemID, remoteClient);
  2248. return UUID.Zero;
  2249. }
  2250. return RezSingleAttachment(att, remoteClient, itemID, AttachmentPt);
  2251. }
  2252. public UUID RezSingleAttachment(SceneObjectGroup att,
  2253. IClientAPI remoteClient, UUID itemID, uint AttachmentPt)
  2254. {
  2255. if (!att.IsDeleted)
  2256. AttachmentPt = att.RootPart.AttachmentPoint;
  2257. ScenePresence presence;
  2258. if (TryGetAvatar(remoteClient.AgentId, out presence))
  2259. {
  2260. presence.Appearance.SetAttachment((int)AttachmentPt, itemID, att.UUID);
  2261. IAvatarFactory ava = RequestModuleInterface<IAvatarFactory>();
  2262. if (ava != null)
  2263. {
  2264. ava.UpdateDatabase(remoteClient.AgentId, presence.Appearance);
  2265. }
  2266. }
  2267. return att.UUID;
  2268. }
  2269. public void RezMultipleAttachments(IClientAPI remoteClient, RezMultipleAttachmentsFromInvPacket.HeaderDataBlock header,
  2270. RezMultipleAttachmentsFromInvPacket.ObjectDataBlock[] objects)
  2271. {
  2272. foreach (RezMultipleAttachmentsFromInvPacket.ObjectDataBlock obj in objects)
  2273. {
  2274. RezSingleAttachment(remoteClient, obj.ItemID, obj.AttachmentPt);
  2275. }
  2276. }
  2277. public void AttachObject(IClientAPI controllingClient, uint localID, uint attachPoint, Quaternion rot, Vector3 pos, bool silent)
  2278. {
  2279. m_sceneGraph.AttachObject(controllingClient, localID, attachPoint, rot, pos, silent);
  2280. }
  2281. public void AttachObject(IClientAPI remoteClient, uint AttachmentPt, UUID itemID, SceneObjectGroup att)
  2282. {
  2283. if (UUID.Zero == itemID)
  2284. {
  2285. m_log.Error("[SCENE INVENTORY]: Unable to save attachment. Error inventory item ID.");
  2286. return;
  2287. }
  2288. if (0 == AttachmentPt)
  2289. {
  2290. m_log.Error("[SCENE INVENTORY]: Unable to save attachment. Error attachment point.");
  2291. return;
  2292. }
  2293. if (null == att.RootPart)
  2294. {
  2295. m_log.Error("[SCENE INVENTORY]: Unable to save attachment for a prim without the rootpart!");
  2296. return;
  2297. }
  2298. ScenePresence presence;
  2299. if (TryGetAvatar(remoteClient.AgentId, out presence))
  2300. {
  2301. presence.Appearance.SetAttachment((int)AttachmentPt, itemID, att.UUID);
  2302. IAvatarFactory ava = RequestModuleInterface<IAvatarFactory>();
  2303. if (ava != null)
  2304. {
  2305. m_log.InfoFormat("[SCENE INVENTORY]: Saving avatar attachment. AgentID:{0} ItemID:{1} AttachmentPoint:{2}", remoteClient.AgentId, itemID, AttachmentPt);
  2306. ava.UpdateDatabase(remoteClient.AgentId, presence.Appearance);
  2307. }
  2308. }
  2309. }
  2310. public void DetachSingleAttachmentToGround(UUID itemID, IClientAPI remoteClient)
  2311. {
  2312. SceneObjectPart part = GetSceneObjectPart(itemID);
  2313. if (part == null || part.ParentGroup == null)
  2314. return;
  2315. UUID inventoryID = part.ParentGroup.GetFromAssetID();
  2316. ScenePresence presence;
  2317. if (TryGetAvatar(remoteClient.AgentId, out presence))
  2318. {
  2319. if (!Permissions.CanRezObject(part.ParentGroup.Children.Count, remoteClient.AgentId, presence.AbsolutePosition))
  2320. return;
  2321. presence.Appearance.DetachAttachment(itemID);
  2322. IAvatarFactory ava = RequestModuleInterface<IAvatarFactory>();
  2323. if (ava != null)
  2324. {
  2325. ava.UpdateDatabase(remoteClient.AgentId, presence.Appearance);
  2326. }
  2327. part.ParentGroup.DetachToGround();
  2328. CachedUserInfo userInfo =
  2329. CommsManager.UserProfileCacheService.GetUserDetails(remoteClient.AgentId);
  2330. if (userInfo != null)
  2331. {
  2332. userInfo.DeleteItem(inventoryID);
  2333. remoteClient.SendRemoveInventoryItem(inventoryID);
  2334. }
  2335. }
  2336. }
  2337. public void DetachSingleAttachmentToInv(UUID itemID, IClientAPI remoteClient)
  2338. {
  2339. ScenePresence presence;
  2340. if (TryGetAvatar(remoteClient.AgentId, out presence))
  2341. {
  2342. presence.Appearance.DetachAttachment(itemID);
  2343. IAvatarFactory ava = RequestModuleInterface<IAvatarFactory>();
  2344. if (ava != null)
  2345. {
  2346. ava.UpdateDatabase(remoteClient.AgentId, presence.Appearance);
  2347. }
  2348. }
  2349. m_sceneGraph.DetachSingleAttachmentToInv(itemID, remoteClient);
  2350. }
  2351. public void GetScriptRunning(IClientAPI controllingClient, UUID objectID, UUID itemID)
  2352. {
  2353. EventManager.TriggerGetScriptRunning(controllingClient, objectID, itemID);
  2354. }
  2355. void ObjectOwner(IClientAPI remoteClient, UUID ownerID, UUID groupID, List<uint> localIDs)
  2356. {
  2357. if (!Permissions.IsGod(remoteClient.AgentId))
  2358. {
  2359. if (ownerID != UUID.Zero)
  2360. return;
  2361. if (!Permissions.CanDeedObject(remoteClient.AgentId, groupID))
  2362. return;
  2363. }
  2364. List<SceneObjectGroup> groups = new List<SceneObjectGroup>();
  2365. foreach (uint localID in localIDs)
  2366. {
  2367. SceneObjectPart part = GetSceneObjectPart(localID);
  2368. if (!groups.Contains(part.ParentGroup))
  2369. groups.Add(part.ParentGroup);
  2370. }
  2371. foreach (SceneObjectGroup sog in groups)
  2372. {
  2373. if (ownerID != UUID.Zero)
  2374. {
  2375. sog.SetOwnerId(ownerID);
  2376. sog.SetGroup(groupID, remoteClient);
  2377. foreach (SceneObjectPart child in sog.Children.Values)
  2378. child.Inventory.ChangeInventoryOwner(ownerID);
  2379. }
  2380. else
  2381. {
  2382. if (!Permissions.CanEditObject(sog.UUID, remoteClient.AgentId))
  2383. continue;
  2384. if (sog.GroupID != groupID)
  2385. continue;
  2386. foreach (SceneObjectPart child in sog.Children.Values)
  2387. {
  2388. child.LastOwnerID = child.OwnerID;
  2389. child.Inventory.ChangeInventoryOwner(groupID);
  2390. }
  2391. sog.SetOwnerId(groupID);
  2392. sog.ApplyNextOwnerPermissions();
  2393. }
  2394. }
  2395. foreach (uint localID in localIDs)
  2396. {
  2397. SceneObjectPart part = GetSceneObjectPart(localID);
  2398. part.GetProperties(remoteClient);
  2399. }
  2400. }
  2401. }
  2402. }