You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

223 lines
4.9 KiB

  1. // Copyright 2013 Beego Authors
  2. // Copyright 2014 The Macaron Authors
  3. //
  4. // Licensed under the Apache License, Version 2.0 (the "License"): you may
  5. // not use this file except in compliance with the License. You may obtain
  6. // a copy of the License at
  7. //
  8. // http://www.apache.org/licenses/LICENSE-2.0
  9. //
  10. // Unless required by applicable law or agreed to in writing, software
  11. // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
  12. // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
  13. // License for the specific language governing permissions and limitations
  14. // under the License.
  15. package session
  16. import (
  17. "container/list"
  18. "fmt"
  19. "sync"
  20. "time"
  21. )
  22. // MemStore represents a in-memory session store implementation.
  23. type MemStore struct {
  24. sid string
  25. lock sync.RWMutex
  26. data map[interface{}]interface{}
  27. lastAccess time.Time
  28. }
  29. // NewMemStore creates and returns a memory session store.
  30. func NewMemStore(sid string) *MemStore {
  31. return &MemStore{
  32. sid: sid,
  33. data: make(map[interface{}]interface{}),
  34. lastAccess: time.Now(),
  35. }
  36. }
  37. // Set sets value to given key in session.
  38. func (s *MemStore) Set(key, val interface{}) error {
  39. s.lock.Lock()
  40. defer s.lock.Unlock()
  41. s.data[key] = val
  42. return nil
  43. }
  44. // Get gets value by given key in session.
  45. func (s *MemStore) Get(key interface{}) interface{} {
  46. s.lock.RLock()
  47. defer s.lock.RUnlock()
  48. return s.data[key]
  49. }
  50. // Delete deletes a key from session.
  51. func (s *MemStore) Delete(key interface{}) error {
  52. s.lock.Lock()
  53. defer s.lock.Unlock()
  54. delete(s.data, key)
  55. return nil
  56. }
  57. // ID returns current session ID.
  58. func (s *MemStore) ID() string {
  59. return s.sid
  60. }
  61. // Release releases resource and save data to provider.
  62. func (_ *MemStore) Release() error {
  63. return nil
  64. }
  65. // Flush deletes all session data.
  66. func (s *MemStore) Flush() error {
  67. s.lock.Lock()
  68. defer s.lock.Unlock()
  69. s.data = make(map[interface{}]interface{})
  70. return nil
  71. }
  72. // MemProvider represents a in-memory session provider implementation.
  73. type MemProvider struct {
  74. lock sync.RWMutex
  75. maxLifetime int64
  76. data map[string]*list.Element
  77. // A priority list whose lastAccess newer gets higer priority.
  78. list *list.List
  79. }
  80. // Init initializes memory session provider.
  81. func (p *MemProvider) Init(maxLifetime int64, _ string) error {
  82. p.lock.Lock()
  83. p.list = list.New()
  84. p.data = make(map[string]*list.Element)
  85. p.maxLifetime = maxLifetime
  86. p.lock.Unlock()
  87. return nil
  88. }
  89. // update expands time of session store by given ID.
  90. func (p *MemProvider) update(sid string) error {
  91. p.lock.Lock()
  92. defer p.lock.Unlock()
  93. if e, ok := p.data[sid]; ok {
  94. e.Value.(*MemStore).lastAccess = time.Now()
  95. p.list.MoveToFront(e)
  96. return nil
  97. }
  98. return nil
  99. }
  100. // Read returns raw session store by session ID.
  101. func (p *MemProvider) Read(sid string) (_ RawStore, err error) {
  102. p.lock.RLock()
  103. e, ok := p.data[sid]
  104. p.lock.RUnlock()
  105. // Only restore if the session is still alive.
  106. if ok && (e.Value.(*MemStore).lastAccess.Unix()+p.maxLifetime) >= time.Now().Unix() {
  107. if err = p.update(sid); err != nil {
  108. return nil, err
  109. }
  110. return e.Value.(*MemStore), nil
  111. }
  112. // Create a new session.
  113. p.lock.Lock()
  114. defer p.lock.Unlock()
  115. if ok {
  116. p.list.Remove(e)
  117. delete(p.data, sid)
  118. }
  119. s := NewMemStore(sid)
  120. p.data[sid] = p.list.PushBack(s)
  121. return s, nil
  122. }
  123. // Exist returns true if session with given ID exists.
  124. func (p *MemProvider) Exist(sid string) bool {
  125. p.lock.RLock()
  126. defer p.lock.RUnlock()
  127. _, ok := p.data[sid]
  128. return ok
  129. }
  130. // Destroy deletes a session by session ID.
  131. func (p *MemProvider) Destroy(sid string) error {
  132. p.lock.Lock()
  133. defer p.lock.Unlock()
  134. e, ok := p.data[sid]
  135. if !ok {
  136. return nil
  137. }
  138. p.list.Remove(e)
  139. delete(p.data, sid)
  140. return nil
  141. }
  142. // Regenerate regenerates a session store from old session ID to new one.
  143. func (p *MemProvider) Regenerate(oldsid, sid string) (RawStore, error) {
  144. if p.Exist(sid) {
  145. return nil, fmt.Errorf("new sid '%s' already exists", sid)
  146. }
  147. s, err := p.Read(oldsid)
  148. if err != nil {
  149. return nil, err
  150. }
  151. if err = p.Destroy(oldsid); err != nil {
  152. return nil, err
  153. }
  154. s.(*MemStore).sid = sid
  155. p.lock.Lock()
  156. defer p.lock.Unlock()
  157. p.data[sid] = p.list.PushBack(s)
  158. return s, nil
  159. }
  160. // Count counts and returns number of sessions.
  161. func (p *MemProvider) Count() int {
  162. return p.list.Len()
  163. }
  164. // GC calls GC to clean expired sessions.
  165. func (p *MemProvider) GC() {
  166. p.lock.RLock()
  167. for {
  168. // No session in the list.
  169. e := p.list.Back()
  170. if e == nil {
  171. break
  172. }
  173. if (e.Value.(*MemStore).lastAccess.Unix() + p.maxLifetime) < time.Now().Unix() {
  174. p.lock.RUnlock()
  175. p.lock.Lock()
  176. p.list.Remove(e)
  177. delete(p.data, e.Value.(*MemStore).sid)
  178. p.lock.Unlock()
  179. p.lock.RLock()
  180. } else {
  181. break
  182. }
  183. }
  184. p.lock.RUnlock()
  185. }
  186. func init() {
  187. Register("memory", &MemProvider{})
  188. }