card 0.0.1 → 1.15.pre

Sign up to get free protection for your applications and to get access to all the features.
Files changed (514) hide show
  1. checksums.yaml +7 -0
  2. data/.rspec +4 -0
  3. data/GPL +331 -0
  4. data/LICENSE +16 -0
  5. data/VERSION +1 -0
  6. data/card.gemspec +48 -19
  7. data/config/initializers/01_init_ruby_extensions.rb +7 -0
  8. data/config/initializers/inflections.rb +13 -0
  9. data/config/initializers/mime_types.rb +13 -0
  10. data/config/initializers/notification.rb +4 -0
  11. data/config/initializers/recaptcha.rb +6 -0
  12. data/config/initializers/uuid_state_file.rb +3 -0
  13. data/db/bootstrap/card_actions.yml +3872 -0
  14. data/db/bootstrap/card_acts.yml +7 -0
  15. data/db/bootstrap/card_changes.yml +11909 -0
  16. data/db/bootstrap/card_references.yml +3970 -0
  17. data/db/bootstrap/cards.yml +10805 -0
  18. data/db/migrate/20110511221913_require_earlier_migrations.rb +19 -0
  19. data/db/migrate/20120105203350_require_1_8_migrations.rb +15 -0
  20. data/db/migrate/20121111025347_require_1_10_migrations.rb +15 -0
  21. data/db/migrate/20121118114000_split_link_type.rb +16 -0
  22. data/db/migrate/20121118115000_update_link_type.rb +19 -0
  23. data/db/migrate/20130106052640_table_cleanup.rb +25 -0
  24. data/db/migrate/20130109015336_trunk_left.rb +14 -0
  25. data/db/migrate/20130411210957_update_codenames.rb +14 -0
  26. data/db/migrate/20140822073704_create_new_revision_tables.rb +43 -0
  27. data/db/migrate/20141001105348_move_revisions_to_actions.rb +62 -0
  28. data/db/migrate/20141121172918_rename_card_migration_table.rb +15 -0
  29. data/db/migrate/20141208132159_remove_present_from_reference_table.rb +9 -0
  30. data/db/migrate/20141216053032_better_index_names.rb +24 -0
  31. data/db/migrate_core_cards/20130411191151_renaming_for_menu.rb +49 -0
  32. data/db/migrate_core_cards/20130411211600_delete_old_related_tab_cards.rb +18 -0
  33. data/db/migrate_core_cards/20130419215612_import_help_text.rb +13 -0
  34. data/db/migrate_core_cards/20130823192433_add_style_cards.rb +87 -0
  35. data/db/migrate_core_cards/20130910183318_move_styles_to_content.rb +11 -0
  36. data/db/migrate_core_cards/20130920214038_jsonize_tinymce.rb +15 -0
  37. data/db/migrate_core_cards/20130920291703_update_stylesheets.rb +18 -0
  38. data/db/migrate_core_cards/20130927191728_account_events.rb +21 -0
  39. data/db/migrate_core_cards/20131016172445_common_css_patch.rb +12 -0
  40. data/db/migrate_core_cards/20140110193325_reset_account_request_type.rb +11 -0
  41. data/db/migrate_core_cards/20140307231621_user_data_to_cards.rb +73 -0
  42. data/db/migrate_core_cards/20140317035504_account_requests_to_signups.rb +51 -0
  43. data/db/migrate_core_cards/20140512155840_add_script_cards.rb +83 -0
  44. data/db/migrate_core_cards/20140629222005_add_email_cards.rb +134 -0
  45. data/db/migrate_core_cards/20140725180118_config_card_updates.rb +9 -0
  46. data/db/migrate_core_cards/20141111083921_delete_machine_output.rb +9 -0
  47. data/db/migrate_core_cards/20141115034214_config_descriptions_etc.rb +14 -0
  48. data/db/migrate_core_cards/20141119001955_make_symlinks_relative.rb +15 -0
  49. data/db/migrate_core_cards/20141120120605_fix_notification_html_message.rb +10 -0
  50. data/db/migrate_core_cards/20141204061304_watchers_to_following.rb +38 -0
  51. data/db/migrate_core_cards/20141208132416_partial_reference_type.rb +7 -0
  52. data/db/migrate_core_cards/20141208162106_add_ace_script.rb +12 -0
  53. data/db/migrate_core_cards/20141216155251_add_more_following_cards.rb +24 -0
  54. data/db/migrate_core_cards/20141230204340_uri_codename.rb +13 -0
  55. data/db/migrate_core_cards/20150202143810_import_bootstrap_layout.rb +17 -0
  56. data/db/migrate_core_cards/20150220134731_following_to_follow_rule.rb +18 -0
  57. data/db/migrate_core_cards/data/1.11_help_text.json +410 -0
  58. data/db/migrate_core_cards/data/1.12_stylesheets/classic_cards.scss +100 -0
  59. data/db/migrate_core_cards/data/1.12_stylesheets/common.scss +226 -0
  60. data/db/migrate_core_cards/data/1.12_stylesheets/right_sidebar.scss +27 -0
  61. data/db/migrate_core_cards/data/1.12_stylesheets/traditional.scss +152 -0
  62. data/db/migrate_core_cards/data/1.13_config_text.json +32 -0
  63. data/db/migrate_core_cards/data/1.14_config_descriptions_etc.json +77 -0
  64. data/db/migrate_core_cards/data/bootstrap_layout.json +81 -0
  65. data/db/migrate_core_cards/data/mailer/follower_notification_email.html +9 -0
  66. data/db/migrate_core_cards/data/mailer/follower_notification_email.txt +11 -0
  67. data/db/migrate_core_cards/data/mailer/mail_config.json +22 -0
  68. data/db/migrate_core_cards/data/mailer/password_reset_email.html +10 -0
  69. data/db/migrate_core_cards/data/mailer/password_reset_email.txt +11 -0
  70. data/db/migrate_core_cards/data/mailer/signup_alert_email.html +7 -0
  71. data/db/migrate_core_cards/data/mailer/signup_alert_email.txt +5 -0
  72. data/db/migrate_core_cards/data/mailer/verification_email.html +9 -0
  73. data/db/migrate_core_cards/data/mailer/verification_email.txt +8 -0
  74. data/db/schema.rb +128 -0
  75. data/db/seeds.rb +8 -0
  76. data/db/version.txt +1 -0
  77. data/db/version_core_cards.txt +1 -0
  78. data/lib/card.rb +46 -3
  79. data/lib/card/active_record_ext.rb +77 -0
  80. data/lib/card/auth.rb +181 -0
  81. data/lib/card/cache.rb +199 -0
  82. data/lib/card/chunk.rb +104 -0
  83. data/lib/card/codename.rb +72 -0
  84. data/lib/card/content.rb +211 -0
  85. data/lib/card/core_ext.rb +105 -0
  86. data/lib/card/core_migration.rb +6 -0
  87. data/lib/card/diff.rb +430 -0
  88. data/lib/card/env.rb +52 -0
  89. data/lib/card/exceptions.rb +44 -0
  90. data/lib/card/format.rb +563 -0
  91. data/lib/card/loader.rb +165 -0
  92. data/lib/card/log.rb +450 -0
  93. data/lib/card/mailer.rb +37 -0
  94. data/lib/card/migration.rb +145 -0
  95. data/lib/card/name.rb +45 -0
  96. data/lib/card/query.rb +81 -0
  97. data/lib/card/query/card_clause.rb +527 -0
  98. data/lib/card/query/clause.rb +21 -0
  99. data/lib/card/query/ref_clause.rb +47 -0
  100. data/lib/card/query/value_clause.rb +65 -0
  101. data/lib/card/reference.rb +63 -0
  102. data/lib/card/set.rb +369 -0
  103. data/lib/card/set_pattern.rb +168 -0
  104. data/lib/card/simplecov_helper.rb +61 -0
  105. data/lib/card/spec_helper.rb +68 -0
  106. data/lib/card/version.rb +9 -2
  107. data/lib/cardio.rb +145 -0
  108. data/lib/generators/card.rb +32 -0
  109. data/lib/generators/card/format/USAGE +9 -0
  110. data/lib/generators/card/format/format_generator.rb +27 -0
  111. data/lib/generators/card/format/templates/format_spec_template.erb +5 -0
  112. data/lib/generators/card/format/templates/format_template.erb +3 -0
  113. data/lib/generators/card/migration/USAGE +24 -0
  114. data/lib/generators/card/migration/migration_generator.rb +38 -0
  115. data/lib/generators/card/migration/templates/card_migration.erb +10 -0
  116. data/lib/generators/card/set/USAGE +15 -0
  117. data/lib/generators/card/set/set_generator.rb +29 -0
  118. data/lib/generators/card/set/templates/set_spec_template.erb +5 -0
  119. data/lib/generators/card/set/templates/set_template.erb +1 -0
  120. data/mod/01_core/chunk/include.rb +85 -0
  121. data/mod/01_core/chunk/link.rb +87 -0
  122. data/mod/01_core/chunk/literal.rb +24 -0
  123. data/mod/01_core/chunk/reference.rb +53 -0
  124. data/mod/01_core/chunk/uri.rb +135 -0
  125. data/mod/01_core/format/data_format.rb +2 -0
  126. data/mod/01_core/format/html_format.rb +129 -0
  127. data/mod/01_core/format/text_format.rb +10 -0
  128. data/mod/01_core/layout/blank.html +5 -0
  129. data/mod/01_core/layout/default.html +20 -0
  130. data/mod/01_core/layout/noside.html +16 -0
  131. data/mod/01_core/layout/pre.html +2 -0
  132. data/mod/01_core/layout/simple.html +5 -0
  133. data/mod/01_core/set/all/active_card.rb +35 -0
  134. data/mod/01_core/set/all/collection.rb +142 -0
  135. data/mod/01_core/set/all/content.rb +64 -0
  136. data/mod/01_core/set/all/erb.rb +11 -0
  137. data/mod/01_core/set/all/fetch.rb +228 -0
  138. data/mod/01_core/set/all/haml.rb +7 -0
  139. data/mod/01_core/set/all/initialize.rb +49 -0
  140. data/mod/01_core/set/all/name.rb +275 -0
  141. data/mod/01_core/set/all/pattern.rb +56 -0
  142. data/mod/01_core/set/all/permissions.rb +284 -0
  143. data/mod/01_core/set/all/phases.rb +185 -0
  144. data/mod/01_core/set/all/references.rb +118 -0
  145. data/mod/01_core/set/all/rules.rb +315 -0
  146. data/mod/01_core/set/all/states.rb +16 -0
  147. data/mod/01_core/set/all/templating.rb +120 -0
  148. data/mod/01_core/set/all/tracked_attributes.rb +114 -0
  149. data/mod/01_core/set/all/trash.rb +50 -0
  150. data/mod/01_core/set/all/type.rb +85 -0
  151. data/mod/01_core/set/all/utils.rb +85 -0
  152. data/mod/01_core/set_pattern/01_all.rb +11 -0
  153. data/mod/01_core/set_pattern/02_all_plus.rb +13 -0
  154. data/mod/01_core/set_pattern/03_type.rb +23 -0
  155. data/mod/01_core/set_pattern/04_star.rb +15 -0
  156. data/mod/01_core/set_pattern/05_rstar.rb +17 -0
  157. data/mod/01_core/set_pattern/06_right.rb +20 -0
  158. data/mod/01_core/set_pattern/07_type_plus_right.rb +23 -0
  159. data/mod/01_core/set_pattern/08_self.rb +19 -0
  160. data/mod/01_core/spec/chunk/literal_spec.rb +14 -0
  161. data/mod/01_core/spec/chunk/uri_spec.rb +292 -0
  162. data/mod/01_core/spec/format/data_format_spec.rb +5 -0
  163. data/mod/01_core/spec/format/html_format_spec.rb +140 -0
  164. data/mod/01_core/spec/format/text_format_spec.rb +5 -0
  165. data/mod/01_core/spec/set/all/active_card_spec.rb +5 -0
  166. data/mod/01_core/spec/set/all/attribute_tracking_spec.rb +21 -0
  167. data/mod/01_core/spec/set/all/collection_spec.rb +65 -0
  168. data/mod/01_core/spec/set/all/content_spec.rb +15 -0
  169. data/mod/01_core/spec/set/all/fetch_spec.rb +204 -0
  170. data/mod/01_core/spec/set/all/initialize_spec.rb +58 -0
  171. data/mod/01_core/spec/set/all/name_spec.rb +61 -0
  172. data/mod/01_core/spec/set/all/pattern_spec.rb +81 -0
  173. data/mod/01_core/spec/set/all/permissions_spec.rb +505 -0
  174. data/mod/01_core/spec/set/all/phases_spec.rb +6 -0
  175. data/mod/01_core/spec/set/all/references_spec.rb +8 -0
  176. data/mod/01_core/spec/set/all/rules2_spec.rb +250 -0
  177. data/mod/01_core/spec/set/all/rules_spec.rb +130 -0
  178. data/mod/01_core/spec/set/all/states_spec.rb +5 -0
  179. data/mod/01_core/spec/set/all/templating_spec.rb +111 -0
  180. data/mod/01_core/spec/set/all/tracked_attributes_spec.rb +328 -0
  181. data/mod/01_core/spec/set/all/trash_spec.rb +34 -0
  182. data/mod/01_core/spec/set/all/type_spec.rb +71 -0
  183. data/mod/01_core/spec/set/all/utils_spec.rb +5 -0
  184. data/mod/01_history/lib/card/act.rb +60 -0
  185. data/mod/01_history/lib/card/action.rb +176 -0
  186. data/mod/01_history/lib/card/change.rb +29 -0
  187. data/mod/01_history/set/all/actions.rb +47 -0
  188. data/mod/01_history/set/all/content_history.rb +131 -0
  189. data/mod/01_history/set/all/history.rb +296 -0
  190. data/mod/02_basic_types/format/css_format.rb +6 -0
  191. data/mod/02_basic_types/format/csv_format.rb +6 -0
  192. data/mod/02_basic_types/format/file_format.rb +3 -0
  193. data/mod/02_basic_types/format/js_format.rb +6 -0
  194. data/mod/02_basic_types/format/json_format.rb +6 -0
  195. data/mod/02_basic_types/format/rss_format.rb +11 -0
  196. data/mod/02_basic_types/format/xml_format.rb +6 -0
  197. data/mod/02_basic_types/set/all/all_css.rb +42 -0
  198. data/mod/02_basic_types/set/all/all_csv.rb +52 -0
  199. data/mod/02_basic_types/set/all/all_js.rb +7 -0
  200. data/mod/02_basic_types/set/all/base.rb +122 -0
  201. data/mod/02_basic_types/set/all/file.rb +13 -0
  202. data/mod/02_basic_types/set/all/json.rb +66 -0
  203. data/mod/02_basic_types/set/all/rss.rb +72 -0
  204. data/mod/02_basic_types/set/all/text.rb +8 -0
  205. data/mod/02_basic_types/set/type/plain_text.rb +10 -0
  206. data/mod/02_basic_types/set/type/pointer.rb +323 -0
  207. data/mod/02_basic_types/spec/set/all/all_css_spec.rb +10 -0
  208. data/mod/02_basic_types/spec/set/all/all_csv_spec.rb +9 -0
  209. data/mod/02_basic_types/spec/set/all/base_spec.rb +57 -0
  210. data/mod/02_basic_types/spec/set/all/file_spec.rb +5 -0
  211. data/mod/02_basic_types/spec/set/all/json_spec.rb +26 -0
  212. data/mod/02_basic_types/spec/set/all/rss_spec.rb +8 -0
  213. data/mod/02_basic_types/spec/set/all/text_spec.rb +5 -0
  214. data/mod/02_basic_types/spec/set/type/plain_text_spec.rb +11 -0
  215. data/mod/02_basic_types/spec/set/type/pointer_spec.rb +103 -0
  216. data/mod/03_machines/lib/card/machine.rb +185 -0
  217. data/mod/03_machines/lib/card/machine_input.rb +40 -0
  218. data/mod/03_machines/lib/javascript/ace.js +18204 -0
  219. data/mod/03_machines/lib/javascript/html5shiv-printshiv.js +1 -0
  220. data/mod/03_machines/lib/javascript/jquery-ui.js +14913 -0
  221. data/mod/03_machines/lib/javascript/jquery.autosize.js +274 -0
  222. data/mod/03_machines/lib/javascript/jquery.fileupload.js +1114 -0
  223. data/mod/03_machines/lib/javascript/jquery.iframe-transport.js +185 -0
  224. data/mod/03_machines/lib/javascript/jquery.js +4 -0
  225. data/mod/03_machines/lib/javascript/jquery.ui.autocomplete.html.js +41 -0
  226. data/mod/03_machines/lib/javascript/jquery_ujs.js +469 -0
  227. data/mod/03_machines/lib/javascript/jquerymobile.js +15454 -0
  228. data/mod/03_machines/lib/javascript/theme-textmate.js +130 -0
  229. data/mod/03_machines/lib/javascript/tinymce.js +13 -0
  230. data/mod/03_machines/lib/javascript/wagn.js.coffee +336 -0
  231. data/mod/03_machines/lib/javascript/wagn_menu.js +72 -0
  232. data/mod/03_machines/lib/javascript/wagn_mod.js.coffee +446 -0
  233. data/mod/03_machines/lib/stylesheets/functional.scss +184 -0
  234. data/mod/03_machines/lib/stylesheets/jquery-ui-smoothness.css +1178 -0
  235. data/mod/03_machines/lib/stylesheets/standard.scss +760 -0
  236. data/mod/03_machines/set/right/machine_output.rb +20 -0
  237. data/mod/03_machines/set/self/script_ace.rb +8 -0
  238. data/mod/03_machines/set/self/script_card_menu.rb +8 -0
  239. data/mod/03_machines/set/self/script_html5shiv_printshiv.rb +7 -0
  240. data/mod/03_machines/set/self/script_jquery.rb +8 -0
  241. data/mod/03_machines/set/self/script_jquery_helper.rb +14 -0
  242. data/mod/03_machines/set/self/script_slot.rb +9 -0
  243. data/mod/03_machines/set/self/script_tinymce.rb +8 -0
  244. data/mod/03_machines/set/self/style_functional.rb +8 -0
  245. data/mod/03_machines/set/self/style_jquery_ui_smoothness.rb +8 -0
  246. data/mod/03_machines/set/self/style_standard.rb +8 -0
  247. data/mod/03_machines/set/type/coffee_script.rb +61 -0
  248. data/mod/03_machines/set/type/css.rb +45 -0
  249. data/mod/03_machines/set/type/java_script.rb +38 -0
  250. data/mod/03_machines/set/type/scss.rb +23 -0
  251. data/mod/03_machines/set/type/skin.rb +12 -0
  252. data/mod/03_machines/spec/lib/shared_machine_examples.rb +175 -0
  253. data/mod/03_machines/spec/lib/shared_machine_input_examples.rb +65 -0
  254. data/mod/03_machines/spec/set/right/machine_output_spec.rb +5 -0
  255. data/mod/03_machines/spec/set/self/style_functional_spec.rb +5 -0
  256. data/mod/03_machines/spec/set/self/style_jquery_ui_smoothness_spec.rb +5 -0
  257. data/mod/03_machines/spec/set/self/style_standard_spec.rb +5 -0
  258. data/mod/03_machines/spec/set/type/coffeescript_spec.rb +29 -0
  259. data/mod/03_machines/spec/set/type/css_spec.rb +34 -0
  260. data/mod/03_machines/spec/set/type/javascript_spec.rb +28 -0
  261. data/mod/03_machines/spec/set/type/scss_spec.rb +56 -0
  262. data/mod/03_machines/spec/set/type/skin_spec.rb +70 -0
  263. data/mod/04_settings/lib/card/setting.rb +65 -0
  264. data/mod/04_settings/set/right/add_help.rb +11 -0
  265. data/mod/04_settings/set/right/comment.rb +91 -0
  266. data/mod/04_settings/set/right/create.rb +3 -0
  267. data/mod/04_settings/set/right/default.rb +3 -0
  268. data/mod/04_settings/set/right/delete.rb +4 -0
  269. data/mod/04_settings/set/right/help.rb +3 -0
  270. data/mod/04_settings/set/right/read.rb +2 -0
  271. data/mod/04_settings/set/right/script.rb +10 -0
  272. data/mod/04_settings/set/right/structure.rb +4 -0
  273. data/mod/04_settings/set/right/style.rb +10 -0
  274. data/mod/04_settings/set/right/update.rb +2 -0
  275. data/mod/04_settings/set/self/accountable.rb +3 -0
  276. data/mod/04_settings/set/self/add_help.rb +2 -0
  277. data/mod/04_settings/set/self/autoname.rb +2 -0
  278. data/mod/04_settings/set/self/captcha.rb +2 -0
  279. data/mod/04_settings/set/self/comment.rb +2 -0
  280. data/mod/04_settings/set/self/create.rb +2 -0
  281. data/mod/04_settings/set/self/default.rb +2 -0
  282. data/mod/04_settings/set/self/delete.rb +2 -0
  283. data/mod/04_settings/set/self/help.rb +2 -0
  284. data/mod/04_settings/set/self/input.rb +2 -0
  285. data/mod/04_settings/set/self/layout.rb +2 -0
  286. data/mod/04_settings/set/self/on_create.rb +2 -0
  287. data/mod/04_settings/set/self/on_delete.rb +2 -0
  288. data/mod/04_settings/set/self/on_update.rb +2 -0
  289. data/mod/04_settings/set/self/options.rb +2 -0
  290. data/mod/04_settings/set/self/options_label.rb +2 -0
  291. data/mod/04_settings/set/self/read.rb +2 -0
  292. data/mod/04_settings/set/self/script.rb +2 -0
  293. data/mod/04_settings/set/self/structure.rb +2 -0
  294. data/mod/04_settings/set/self/style.rb +2 -0
  295. data/mod/04_settings/set/self/table_of_contents.rb +2 -0
  296. data/mod/04_settings/set/self/thanks.rb +2 -0
  297. data/mod/04_settings/set/self/update.rb +2 -0
  298. data/mod/04_settings/set/type/setting.rb +82 -0
  299. data/mod/04_settings/spec/set/right/add_help_spec.rb +5 -0
  300. data/mod/04_settings/spec/set/right/comment_spec.rb +41 -0
  301. data/mod/04_settings/spec/set/right/create_spec.rb +10 -0
  302. data/mod/04_settings/spec/set/right/default_spec.rb +5 -0
  303. data/mod/04_settings/spec/set/right/delete_spec.rb +5 -0
  304. data/mod/04_settings/spec/set/right/help_spec.rb +5 -0
  305. data/mod/04_settings/spec/set/right/read_spec.rb +5 -0
  306. data/mod/04_settings/spec/set/right/script_spec.rb +24 -0
  307. data/mod/04_settings/spec/set/right/structure_spec.rb +17 -0
  308. data/mod/04_settings/spec/set/right/style_spec.rb +29 -0
  309. data/mod/04_settings/spec/set/right/update_spec.rb +5 -0
  310. data/mod/04_settings/spec/set/type/setting_spec.rb +10 -0
  311. data/mod/05_email/format/email_html_format.rb +9 -0
  312. data/mod/05_email/format/email_text_format.rb +11 -0
  313. data/mod/05_email/lib/card/follow_option.rb +90 -0
  314. data/mod/05_email/set/all/email_html.rb +5 -0
  315. data/mod/05_email/set/all/email_text.rb +5 -0
  316. data/mod/05_email/set/all/follow.rb +256 -0
  317. data/mod/05_email/set/all/notify.rb +223 -0
  318. data/mod/05_email/set/all/observer.rb +27 -0
  319. data/mod/05_email/set/right/bcc.rb +45 -0
  320. data/mod/05_email/set/right/cc.rb +3 -0
  321. data/mod/05_email/set/right/follow.rb +9 -0
  322. data/mod/05_email/set/right/follow_fields.rb +3 -0
  323. data/mod/05_email/set/right/followers.rb +30 -0
  324. data/mod/05_email/set/right/following.rb +52 -0
  325. data/mod/05_email/set/right/from.rb +3 -0
  326. data/mod/05_email/set/right/html_message.rb +3 -0
  327. data/mod/05_email/set/right/to.rb +3 -0
  328. data/mod/05_email/set/self/always.rb +14 -0
  329. data/mod/05_email/set/self/created.rb +21 -0
  330. data/mod/05_email/set/self/edited.rb +24 -0
  331. data/mod/05_email/set/self/follow.rb +2 -0
  332. data/mod/05_email/set/self/follow_defaults.rb +66 -0
  333. data/mod/05_email/set/self/never.rb +15 -0
  334. data/mod/05_email/set/type/email_template.rb +113 -0
  335. data/mod/05_email/set/type_plus_right/user/follow.rb +176 -0
  336. data/mod/05_email/spec/set/all/follow_spec.rb +133 -0
  337. data/mod/05_email/spec/set/all/notify_spec.rb +364 -0
  338. data/mod/05_email/spec/set/all/observer_spec.rb +76 -0
  339. data/mod/05_email/spec/set/right/followers_spec.rb +126 -0
  340. data/mod/05_email/spec/set/right/following_spec.rb +4 -0
  341. data/mod/05_email/spec/set/self/follow_defaults_spec.rb +18 -0
  342. data/mod/05_standard/file/103/icon-6566.ico +0 -0
  343. data/mod/05_standard/file/103/large-6566.ico +0 -0
  344. data/mod/05_standard/file/103/medium-6566.ico +0 -0
  345. data/mod/05_standard/file/103/original-6566.ico +0 -0
  346. data/mod/05_standard/file/103/small-6566.ico +0 -0
  347. data/mod/05_standard/file/79/icon-6556.png +0 -0
  348. data/mod/05_standard/file/79/large-6556.png +0 -0
  349. data/mod/05_standard/file/79/medium-6556.png +0 -0
  350. data/mod/05_standard/file/79/original-6556.png +0 -0
  351. data/mod/05_standard/file/79/small-6556.png +0 -0
  352. data/mod/05_standard/file/790/icon-6419.png +0 -0
  353. data/mod/05_standard/file/790/large-6419.png +0 -0
  354. data/mod/05_standard/file/790/medium-6419.png +0 -0
  355. data/mod/05_standard/file/790/original-6419.png +0 -0
  356. data/mod/05_standard/file/790/small-6419.png +0 -0
  357. data/mod/05_standard/set/all/account.rb +67 -0
  358. data/mod/05_standard/set/all/attach.rb +152 -0
  359. data/mod/05_standard/set/all/comment.rb +39 -0
  360. data/mod/05_standard/set/all/error.rb +214 -0
  361. data/mod/05_standard/set/all/event_viz.rb +62 -0
  362. data/mod/05_standard/set/all/links.rb +110 -0
  363. data/mod/05_standard/set/all/rich_html/content.rb +173 -0
  364. data/mod/05_standard/set/all/rich_html/editing.rb +145 -0
  365. data/mod/05_standard/set/all/rich_html/form.rb +234 -0
  366. data/mod/05_standard/set/all/rich_html/header.rb +64 -0
  367. data/mod/05_standard/set/all/rich_html/wrapper.rb +105 -0
  368. data/mod/05_standard/set/right/account.rb +180 -0
  369. data/mod/05_standard/set/right/email.rb +52 -0
  370. data/mod/05_standard/set/right/password.rb +39 -0
  371. data/mod/05_standard/set/right/salt.rb +5 -0
  372. data/mod/05_standard/set/right/stats.rb +35 -0
  373. data/mod/05_standard/set/right/status.rb +9 -0
  374. data/mod/05_standard/set/right/token.rb +5 -0
  375. data/mod/05_standard/set/right/when_created.rb +5 -0
  376. data/mod/05_standard/set/right/when_last_edited.rb +5 -0
  377. data/mod/05_standard/set/rstar/rules.rb +407 -0
  378. data/mod/05_standard/set/self/account_links.rb +61 -0
  379. data/mod/05_standard/set/self/alerts.rb +5 -0
  380. data/mod/05_standard/set/self/all.rb +21 -0
  381. data/mod/05_standard/set/self/foot.rb +8 -0
  382. data/mod/05_standard/set/self/head.rb +121 -0
  383. data/mod/05_standard/set/self/navbox.rb +20 -0
  384. data/mod/05_standard/set/self/now.rb +6 -0
  385. data/mod/05_standard/set/self/recent.rb +48 -0
  386. data/mod/05_standard/set/self/search.rb +43 -0
  387. data/mod/05_standard/set/self/signin.rb +121 -0
  388. data/mod/05_standard/set/self/stats.rb +91 -0
  389. data/mod/05_standard/set/self/version.rb +9 -0
  390. data/mod/05_standard/set/type/basic.rb +46 -0
  391. data/mod/05_standard/set/type/cardtype.rb +43 -0
  392. data/mod/05_standard/set/type/date.rb +8 -0
  393. data/mod/05_standard/set/type/file.rb +85 -0
  394. data/mod/05_standard/set/type/html.rb +22 -0
  395. data/mod/05_standard/set/type/image.rb +67 -0
  396. data/mod/05_standard/set/type/layout_type.rb +14 -0
  397. data/mod/05_standard/set/type/number.rb +19 -0
  398. data/mod/05_standard/set/type/phrase.rb +3 -0
  399. data/mod/05_standard/set/type/search_type.rb +243 -0
  400. data/mod/05_standard/set/type/set.rb +198 -0
  401. data/mod/05_standard/set/type/signup.rb +144 -0
  402. data/mod/05_standard/set/type/toggle.rb +12 -0
  403. data/mod/05_standard/set/type/uri.rb +11 -0
  404. data/mod/05_standard/set/type/user.rb +72 -0
  405. data/mod/05_standard/spec/chunk/include_spec.rb +186 -0
  406. data/mod/05_standard/spec/chunk/link_spec.rb +61 -0
  407. data/mod/05_standard/spec/format/css_format_spec.rb +5 -0
  408. data/mod/05_standard/spec/format/csv_format_spec.rb +5 -0
  409. data/mod/05_standard/spec/format/email_html_format_spec.rb +5 -0
  410. data/mod/05_standard/spec/format/file_format_spec.rb +5 -0
  411. data/mod/05_standard/spec/format/js_format_spec.rb +5 -0
  412. data/mod/05_standard/spec/format/json_format_spec.rb +5 -0
  413. data/mod/05_standard/spec/format/rss_format_spec.rb +5 -0
  414. data/mod/05_standard/spec/format/xml_format_spec.rb +5 -0
  415. data/mod/05_standard/spec/set/all/account_spec.rb +137 -0
  416. data/mod/05_standard/spec/set/all/attach_spec.rb +9 -0
  417. data/mod/05_standard/spec/set/all/comment_spec.rb +5 -0
  418. data/mod/05_standard/spec/set/all/email_html_spec.rb +15 -0
  419. data/mod/05_standard/spec/set/all/error_spec.rb +9 -0
  420. data/mod/05_standard/spec/set/all/event_viz_spec.rb +9 -0
  421. data/mod/05_standard/spec/set/all/history_spec.rb +173 -0
  422. data/mod/05_standard/spec/set/all/rich_html/form_spec.rb +43 -0
  423. data/mod/05_standard/spec/set/all/rich_html/wrapper_spec.rb +28 -0
  424. data/mod/05_standard/spec/set/right/account_spec.rb +162 -0
  425. data/mod/05_standard/spec/set/right/email_spec.rb +55 -0
  426. data/mod/05_standard/spec/set/right/password_spec.rb +45 -0
  427. data/mod/05_standard/spec/set/right/salt_spec.rb +5 -0
  428. data/mod/05_standard/spec/set/right/stats_spec.rb +13 -0
  429. data/mod/05_standard/spec/set/right/status_spec.rb +5 -0
  430. data/mod/05_standard/spec/set/right/token_spec.rb +5 -0
  431. data/mod/05_standard/spec/set/right/when_created_spec.rb +7 -0
  432. data/mod/05_standard/spec/set/right/when_last_edited_spec.rb +7 -0
  433. data/mod/05_standard/spec/set/rstar/rules_spec.rb +25 -0
  434. data/mod/05_standard/spec/set/self/account_links_spec.rb +9 -0
  435. data/mod/05_standard/spec/set/self/alerts_spec.rb +5 -0
  436. data/mod/05_standard/spec/set/self/all_spec.rb +48 -0
  437. data/mod/05_standard/spec/set/self/foot_spec.rb +5 -0
  438. data/mod/05_standard/spec/set/self/head_spec.rb +17 -0
  439. data/mod/05_standard/spec/set/self/navbox_spec.rb +7 -0
  440. data/mod/05_standard/spec/set/self/now_spec.rb +7 -0
  441. data/mod/05_standard/spec/set/self/recent_spec.rb +5 -0
  442. data/mod/05_standard/spec/set/self/search_spec.rb +5 -0
  443. data/mod/05_standard/spec/set/self/signin_spec.rb +73 -0
  444. data/mod/05_standard/spec/set/self/stats_spec.rb +12 -0
  445. data/mod/05_standard/spec/set/self/version_spec.rb +7 -0
  446. data/mod/05_standard/spec/set/type/basic_spec.rb +5 -0
  447. data/mod/05_standard/spec/set/type/cardtype_spec.rb +5 -0
  448. data/mod/05_standard/spec/set/type/date_spec.rb +7 -0
  449. data/mod/05_standard/spec/set/type/email_template_spec.rb +208 -0
  450. data/mod/05_standard/spec/set/type/file_spec.rb +5 -0
  451. data/mod/05_standard/spec/set/type/html_spec.rb +23 -0
  452. data/mod/05_standard/spec/set/type/image_spec.rb +16 -0
  453. data/mod/05_standard/spec/set/type/layout_type_spec.rb +7 -0
  454. data/mod/05_standard/spec/set/type/number_spec.rb +7 -0
  455. data/mod/05_standard/spec/set/type/phrase_spec.rb +7 -0
  456. data/mod/05_standard/spec/set/type/search_type_spec.rb +27 -0
  457. data/mod/05_standard/spec/set/type/set_spec.rb +26 -0
  458. data/mod/05_standard/spec/set/type/signup_spec.rb +228 -0
  459. data/mod/05_standard/spec/set/type/toggle_spec.rb +12 -0
  460. data/mod/05_standard/spec/set/type/uri_spec.rb +41 -0
  461. data/mod/05_standard/spec/set/type/user_spec.rb +5 -0
  462. data/mod/06_bootstrap/lib/javascript/bootstrap.js +2306 -0
  463. data/mod/06_bootstrap/lib/javascript/bootstrap.min.js +7 -0
  464. data/mod/06_bootstrap/lib/stylesheets/bootstrap-theme.css +476 -0
  465. data/mod/06_bootstrap/lib/stylesheets/bootstrap.css +6565 -0
  466. data/mod/06_bootstrap/lib/stylesheets/darkly.css +6583 -0
  467. data/mod/06_bootstrap/set/all/bootstrap/form.rb +37 -0
  468. data/mod/06_bootstrap/set/all/bootstrap/header.rb +29 -0
  469. data/mod/06_bootstrap/set/all/bootstrap/wrapper.rb +12 -0
  470. data/mod/06_bootstrap/set/all/rich_bootstrap.rb +28 -0
  471. data/mod/06_bootstrap/set/self/bootstrap_css.rb +7 -0
  472. data/mod/06_bootstrap/set/self/bootstrap_js.rb +7 -0
  473. data/mod/06_bootstrap/set/self/bootstrap_theme_css.rb +7 -0
  474. data/mod/06_bootstrap/set/self/navbox.rb +32 -0
  475. data/mod/06_bootstrap/set/type/search_type.rb +79 -0
  476. data/mod/06_bootstrap/spec/set/all/bootstrap/form_spec.rb +13 -0
  477. data/spec/lib/card/action_spec.rb +14 -0
  478. data/spec/lib/card/auth_spec.rb +17 -0
  479. data/spec/lib/card/cache_spec.rb +122 -0
  480. data/spec/lib/card/chunk_spec.rb +17 -0
  481. data/spec/lib/card/codename_spec.rb +25 -0
  482. data/spec/lib/card/content_spec.rb +314 -0
  483. data/spec/lib/card/diff_spec.rb +210 -0
  484. data/spec/lib/card/format_spec.rb +82 -0
  485. data/spec/lib/card/loader_spec.rb +39 -0
  486. data/spec/lib/card/log_spec.rb +270 -0
  487. data/spec/lib/card/name_spec.rb +279 -0
  488. data/spec/lib/card/query_spec.rb +456 -0
  489. data/spec/lib/card/reference_spec.rb +213 -0
  490. data/spec/lib/card/set_pattern_spec.rb +56 -0
  491. data/spec/lib/card/set_spec.rb +88 -0
  492. data/spec/mailers/mailer_spec.rb +64 -0
  493. data/spec/models/card/cardtype_spec.rb +216 -0
  494. data/spec/models/card/create_spec.rb +82 -0
  495. data/spec/models/card/trash_spec.rb +260 -0
  496. data/spec/models/card/type_transition_spec.rb +161 -0
  497. data/spec/models/card/validation_spec.rb +36 -0
  498. data/spec/models/card_spec.rb +177 -0
  499. data/spec/spec_helper.rb +130 -0
  500. data/test/fixtures/.gitkeep +0 -0
  501. data/test/fixtures/card_actions.yml +5706 -0
  502. data/test/fixtures/card_acts.yml +835 -0
  503. data/test/fixtures/card_changes.yml +17182 -0
  504. data/test/fixtures/card_references.yml +5405 -0
  505. data/test/fixtures/cards.yml +15533 -0
  506. data/test/fixtures/mao2.jpg +0 -0
  507. data/test/fixtures/rails.gif +0 -0
  508. data/test/seed.rb +201 -0
  509. metadata +762 -38
  510. data/.gitignore +0 -17
  511. data/Gemfile +0 -4
  512. data/LICENSE.txt +0 -22
  513. data/README.md +0 -29
  514. data/Rakefile +0 -1
@@ -0,0 +1,165 @@
1
+ # -*- encoding : utf-8 -*-
2
+
3
+ require_dependency 'card/set'
4
+ require_dependency 'card/set_pattern'
5
+
6
+ class Card
7
+ class << self
8
+ def config
9
+ Cardio.config
10
+ end
11
+
12
+ def paths
13
+ Cardio.paths
14
+ end
15
+ end
16
+
17
+ module Loader
18
+
19
+ class << self
20
+ def load_mods
21
+ load_set_patterns
22
+ load_formats
23
+ load_sets
24
+
25
+ if Wagn.config.performance_logger
26
+ Card::Log::Performance.load_config Wagn.config.performance_logger
27
+ end
28
+ end
29
+
30
+ def load_chunks
31
+ mod_dirs.each do |mod|
32
+ load_dir "#{mod}/chunk/*.rb"
33
+ end
34
+ end
35
+
36
+ def load_layouts
37
+ mod_dirs.inject({}) do |hash, mod|
38
+ dirname = "#{mod}/layout"
39
+ if File.exists? dirname
40
+ Dir.foreach( dirname ) do |filename|
41
+ next if filename =~ /^\./
42
+ hash[ filename.gsub /\.html$/, '' ] = File.read( [dirname, filename] * '/' )
43
+ end
44
+ end
45
+ hash
46
+ end
47
+ end
48
+
49
+ def mod_dirs
50
+ @@mod_dirs ||= begin
51
+ if Card.paths['local-mod']
52
+ Card.paths['mod'] << Card.paths['local-mod']
53
+ Rails.logger.warn 'DEPRECATION WARNING: Append to paths[\'mod\'] vs. local-mod for configuring location of local (deck) modules.'
54
+ end
55
+
56
+ Card.paths['mod'].existent.map do |dirname|
57
+ Dir.entries( dirname ).sort.map do |filename|
58
+ "#{dirname}/#{filename}" if filename !~ /^\./
59
+ end.compact
60
+ end.flatten.compact
61
+ end
62
+ end
63
+
64
+ private
65
+
66
+ def load_set_patterns
67
+ if rewrite_tmp_files?
68
+ load_set_patterns_from_source
69
+ end
70
+ load_dir "#{Card.paths['tmp/set_pattern'].first}/*.rb"
71
+ end
72
+
73
+ def load_set_patterns_from_source
74
+ prepare_tmp_dir 'tmp/set_pattern'
75
+ seq = 100
76
+ mod_dirs.each do |mod|
77
+ dirname = "#{mod}/set_pattern"
78
+ if Dir.exists? dirname
79
+ Dir.entries( dirname ).sort.each do |filename|
80
+ if m = filename.match( /^(\d+_)?([^\.]*).rb/) and key = m[2]
81
+ filename = [ dirname, filename ] * '/'
82
+ SetPattern.write_tmp_file key, filename, seq
83
+ seq = seq + 1
84
+ end
85
+ end
86
+ end
87
+ end
88
+ end
89
+
90
+ def load_formats
91
+ #cheating on load issues now by putting all inherited-from formats in core mod.
92
+ mod_dirs.each do |mod|
93
+ load_dir "#{mod}/format/*.rb"
94
+ end
95
+ end
96
+
97
+ def load_sets
98
+ prepare_tmp_dir 'tmp/set'
99
+ load_sets_by_pattern
100
+ Set.process_base_modules
101
+ Set.clean_empty_modules
102
+ end
103
+
104
+
105
+ def load_sets_by_pattern
106
+ Card.set_patterns.reverse.map(&:pattern_code).each do |set_pattern|
107
+ pattern_tmp_dir = "#{Card.paths['tmp/set'].first}/#{set_pattern}"
108
+ if rewrite_tmp_files?
109
+ Dir.mkdir pattern_tmp_dir
110
+ load_implicit_sets_from_source set_pattern
111
+ end
112
+ if Dir.exists? pattern_tmp_dir
113
+ load_dir "#{pattern_tmp_dir}/**/*.rb"
114
+ end
115
+ end
116
+ end
117
+
118
+ def load_implicit_sets_from_source set_pattern
119
+ seq = 1000
120
+ mod_dirs.each do |mod_dir|
121
+ dirname = [mod_dir, 'set', set_pattern] * '/'
122
+ next unless File.exists?( dirname )
123
+
124
+ old_pwd = Dir.pwd
125
+ Dir.chdir dirname
126
+ Dir.glob( "**/*.rb" ).sort.each do |anchor_path|
127
+ path_parts = anchor_path.gsub(/\.rb/,'').split(File::SEPARATOR)
128
+ filename = File.join dirname, anchor_path
129
+ Set.write_tmp_file set_pattern, path_parts, filename, seq
130
+ seq = seq + 1
131
+ end
132
+ Dir.chdir old_pwd
133
+ end
134
+ end
135
+
136
+ def prepare_tmp_dir path
137
+ if rewrite_tmp_files?
138
+ p = Card.paths[ path ]
139
+ if p.existent.first
140
+ FileUtils.rm_rf p.first, :secure=>true
141
+ end
142
+ Dir.mkdir p.first
143
+ end
144
+ end
145
+
146
+ def rewrite_tmp_files?
147
+ if defined?( @@rewrite )
148
+ @@rewrite
149
+ else
150
+ @@rewrite = !( Rails.env.production? and Card.paths['tmp/set'].existent.first )
151
+ end
152
+ end
153
+
154
+ def load_dir dir
155
+ Dir[dir].sort.each do |file|
156
+ # puts Benchmark.measure("from #load_dir: rd: #{file}") {
157
+ require_dependency file
158
+ # }.format("%n: %t %r")
159
+ end
160
+ end
161
+ end
162
+ end
163
+
164
+ end
165
+
data/lib/card/log.rb ADDED
@@ -0,0 +1,450 @@
1
+ require 'csv'
2
+
3
+ class Card::Log
4
+
5
+ class Request
6
+
7
+ def self.path
8
+ path = (Card.paths['request_log'] && Card.paths['request_log'].first) || File.dirname(Card.paths['log'].first)
9
+ filename = "#{Date.today}_#{Rails.env}.csv"
10
+ File.join path, filename
11
+ end
12
+
13
+ def self.write_log_entry controller
14
+ return if controller.env["REQUEST_URI"] =~ %r{^/files?/}
15
+
16
+ controller.instance_eval do
17
+ log = []
18
+ log << (Card::Env.ajax? ? "YES" : "NO")
19
+ log << env["REMOTE_ADDR"]
20
+ log << Card::Auth.current_id
21
+ log << card.name
22
+ log << action_name
23
+ log << params['view'] || (s = params['success'] and s['view'])
24
+ log << env["REQUEST_METHOD"]
25
+ log << status
26
+ log << env["REQUEST_URI"]
27
+ log << DateTime.now.to_s
28
+ log << env['HTTP_ACCEPT_LANGUAGE'].to_s.scan(/^[a-z]{2}/).first
29
+ log << env["HTTP_REFERER"]
30
+
31
+ File.open(Card::Log::Request.path, "a") do |f|
32
+ f.write CSV.generate_line(log)
33
+ end
34
+ end
35
+ end
36
+
37
+ end
38
+
39
+
40
+ class Performance
41
+ # To enable logging add a performance_logger hash to your configuration and change the log_level to :wagn
42
+ # config options
43
+ #
44
+ # Example:
45
+ # config.performance_logger = {
46
+ # :min_time => 100, # show only method calls that are slower than 100ms
47
+ # :max_depth => 3, # show nested method calls only up to depth 3
48
+ # :details=> true # show method arguments and sql
49
+ # :methods => [:event, :search, :fetch, :view], # choose methods to log
50
+ # }
51
+ #
52
+ # If you give :methods a hash you can log arbitrary methods. The syntax is as follows:
53
+ # class => method type => method name => log options
54
+ #
55
+ # Example:
56
+ # Card => {
57
+ # :instance => [ :fetch, :search ],
58
+ # :singleton => { :fetch => { :title => 'Card.fetch' } },
59
+ # :all => {
60
+ # :fetch => {
61
+ # :message => 2 # use second argument passed to fetch
62
+ # :details => :to_s # use return value of to_s in method context
63
+ # :title => proc { |method_context| method_context.name }
64
+ # },
65
+ # },
66
+ # },
67
+ #
68
+ # class, method type and log options are optional.
69
+ # Default values are 'Card', ':all' and { :title => method name, :message => first argument, :details=> remaining arguments }.
70
+ # For example [:fetch] is equivalent to Card => { :all => { :fetch => { :message=>1, :details=>1..-1 } }
71
+
72
+ DEFAULT_CLASS = Card
73
+ DEFAULT_METHOD_TYPE = :all
74
+ DEFAULT_METHOD_OPTIONS = {
75
+ :title => :method_name,
76
+ :message => 1,
77
+ :details => 1..-1,
78
+ :context => nil
79
+ }
80
+
81
+ SPECIAL_METHODS = [:search, :view, :event] # these methods have already a Wagn.with_logging block
82
+ # we don't have to monkey patch them, only turn the logging on with adding the symbol to the methods hash
83
+
84
+
85
+
86
+ TAB_SIZE = 3
87
+ @@log = []
88
+ @@context_entries = []
89
+ @@active_entries = []
90
+ @@current_level = 0
91
+
92
+
93
+ class << self
94
+ def load_config args
95
+ @details = args[:details] || false
96
+ @max_depth = args[:max_depth] || false
97
+ @min_time = args[:min_time] || false
98
+ @enabled_methods = ::Set.new
99
+ prepare_methods_for_logging args[:methods] if args[:methods]
100
+ end
101
+
102
+ def start args={}
103
+ @@current_level = 0
104
+ @@log = []
105
+ @@context_entries = []
106
+ @@active_entries = []
107
+ @@first_entry = new_entry(args)
108
+ end
109
+
110
+ def stop
111
+ while (entry = @@context_entries.pop) do
112
+ finish_entry entry
113
+ end
114
+ if @@first_entry
115
+ @@first_entry.save_duration
116
+ finish_entry @@first_entry
117
+ end
118
+ print_log
119
+ end
120
+
121
+
122
+ def with_timer method, args, &block
123
+ if args[:context]
124
+
125
+ # if the previous context was created by an entry on the same level
126
+ # then finish the current context if it's a different context
127
+ if @@context_entries.last && @@current_level == @@context_entries.last.level+1 &&
128
+ args[:context] != @@context_entries.last.context
129
+ finish_entry @@context_entries.pop
130
+ end
131
+
132
+ # start new context if it's different from the parent context
133
+ if @@context_entries.empty? || args[:context] != @@context_entries.last.context
134
+ @@context_entries << new_entry( :title=>'process', :message=>args[:context], :context=>args[:context] )
135
+ end
136
+ end
137
+
138
+ timer = new_entry args.merge(:method=>method )
139
+ begin
140
+ result = block.call
141
+ ensure
142
+ timer.save_duration
143
+ finish_entry timer
144
+
145
+ # finish all deeper nested contexts
146
+ while @@context_entries.last && @@context_entries.last.level >= @@current_level
147
+ finish_entry @@context_entries.pop
148
+ end
149
+ # we don't know whether the next entry will belong to the same context or will start a new one
150
+ # so we save the time
151
+ @@context_entries.last.save_duration if @@context_entries.last
152
+ end
153
+ result
154
+ end
155
+
156
+
157
+ def enable_method method_name
158
+ @enabled_methods ||= ::Set.new
159
+ @enabled_methods << method_name
160
+ end
161
+
162
+ def enabled_method? method_name
163
+ @enabled_methods && @enabled_methods.include?(method_name)
164
+ end
165
+
166
+ private
167
+
168
+ def print_log
169
+ @@log.each do |entry|
170
+ Rails.logger.wagn entry.to_s! if entry.valid
171
+ end
172
+ end
173
+
174
+ def new_entry args
175
+ args.delete(:details) unless @details
176
+ level = @@current_level
177
+
178
+ last_entry = @@active_entries.last
179
+ parent = if last_entry
180
+ last_entry.level == level ? last_entry.parent : last_entry
181
+ end
182
+
183
+ @@log << Card::Log::Performance::Entry.new(parent, level, args )
184
+ @@current_level += 1
185
+ @@active_entries << @@log.last
186
+
187
+ @@log.last
188
+ end
189
+
190
+ def finish_entry entry
191
+ if (@max_depth && entry.level > @max_depth) || (@min_time && entry.duration < @min_time)
192
+ entry.delete
193
+ end
194
+ @@active_entries.pop
195
+ @@current_level -= 1
196
+ end
197
+
198
+ def prepare_methods_for_logging args
199
+ classes = hashify_and_verify_keys( args, DEFAULT_CLASS ) do |key|
200
+ key.kind_of?(Class) || key.kind_of?(Module)
201
+ end
202
+
203
+ classes.each do |klass, method_types|
204
+ klass.extend BigBrother # add watch methods
205
+
206
+ method_types = hashify_and_verify_keys( method_types, DEFAULT_METHOD_TYPE ) do |key|
207
+ [:all, :instance, :singleton].include? key
208
+ end
209
+
210
+ method_types.each do |method_type, methods|
211
+ methods = hashify_and_verify_keys methods
212
+ methods.each do |method_name, options|
213
+ klass.watch_method method_name, method_type, DEFAULT_METHOD_OPTIONS.merge(options)
214
+ end
215
+ end
216
+
217
+ end
218
+ end
219
+
220
+
221
+ def hashify_and_verify_keys args, default_key=nil
222
+ if default_key
223
+ case args
224
+ when Symbol
225
+ { default_key => [ args ] }
226
+ when Array
227
+ { default_key => args }
228
+ when Hash
229
+ if block_given?
230
+ args.keys.select{ |key| !(yield(key)) }.each do |key|
231
+ args[default_key] = { key => args[key] }
232
+ args.delete key
233
+ end
234
+ end
235
+ args
236
+ end
237
+ else
238
+ case args
239
+ when Symbol
240
+ { args => {} }
241
+ when Array
242
+ args.inject({}) do |h, key|
243
+ h[key] = {}
244
+ h
245
+ end
246
+ else
247
+ args
248
+ end
249
+ end
250
+ end
251
+
252
+ end
253
+
254
+
255
+ class Entry
256
+ attr_accessor :level, :valid, :context, :parent, :children_cnt, :duration
257
+
258
+ def initialize( parent, level, args )
259
+ @start = Time.new
260
+ @message = "#{ args[:title] || args[:method] || '' }"
261
+ @message += ": #{ args[:message] }" if args[:message]
262
+ @details = args[:details]
263
+ @context = args[:context]
264
+ @level = level
265
+ @duration = nil
266
+ @valid = true
267
+ @parent = parent
268
+ @children_cnt = 0
269
+ if @parent
270
+ @parent.add_children
271
+ #@sibling_nr = @parent.children_cnt
272
+ end
273
+ end
274
+
275
+ def add_children
276
+ @children_cnt += 1
277
+ end
278
+
279
+ def delete_children
280
+ @children_cnt -= 1
281
+ end
282
+
283
+ def has_younger_siblings?
284
+ @parent && @parent.children_cnt > 0 #@sibling_nr
285
+ end
286
+
287
+ def save_duration
288
+ @duration = (Time.now - @start) * 1000
289
+ end
290
+
291
+ def delete
292
+ @valid = false
293
+ @parent.delete_children if @parent
294
+ end
295
+
296
+
297
+ # deletes the children counts in order to print the tree;
298
+ # must be called in the right order
299
+ #
300
+ # More robuts but more expensive approach: use @sibling_nr instead of counting @children_cnt down,
301
+ # but @sibling_nr has to be updated for all siblings of an entry if the entry gets deleted due to
302
+ # min_time or max_depth restrictions in the config, so we have to save all children relations for that
303
+ def to_s!
304
+ @to_s ||= begin
305
+ msg = indent
306
+ msg += "(%d.2ms) " % @duration if @duration
307
+ msg += @message if @message
308
+
309
+ if @details
310
+ msg += ", " + @details.to_s.gsub( "\n", "\n#{ indent(false) }#{' '* TAB_SIZE}" )
311
+ end
312
+ @parent.delete_children if @parent
313
+ msg
314
+ end
315
+ end
316
+
317
+ private
318
+
319
+ def indent link=true
320
+ @indent ||= begin
321
+ if @level == 0
322
+ "\n"
323
+ else
324
+ res = ' '
325
+ res += (1..level-1).inject('') do |msg, index|
326
+ if younger_siblings[index]
327
+ msg << '|' + ' ' * (TAB_SIZE-1)
328
+ else
329
+ msg << ' ' * TAB_SIZE
330
+ end
331
+ end
332
+
333
+ res += link ? '|--' : ' '
334
+ end
335
+ end
336
+ end
337
+
338
+ def younger_siblings
339
+ res = []
340
+ next_parent = self
341
+ while (next_parent)
342
+ res << next_parent.has_younger_siblings?
343
+ next_parent = next_parent.parent
344
+ end
345
+ res.reverse
346
+ end
347
+
348
+ end
349
+
350
+
351
+ module BigBrother
352
+
353
+ def watch_method method_name, method_type=:all, options={}
354
+ Card::Log::Performance.enable_method method_name
355
+
356
+ if !SPECIAL_METHODS.include? method_name
357
+ if method_type == :all || method_type == :singleton
358
+ add_singleton_logging method_name, options
359
+ end
360
+ if method_type == :all || method_type == :instance
361
+ add_instance_logging method_name, options
362
+ end
363
+ end
364
+ end
365
+
366
+ def watch_instance_method *names
367
+ names.each do |name|
368
+ watch_method name, :instance
369
+ end
370
+ end
371
+
372
+ def watch_singleton_method *names
373
+ names.each do |name|
374
+ watch_method name, :singleton
375
+ end
376
+ end
377
+
378
+ def watch_all_instance_methods
379
+ watch_instance_method *instance_methods
380
+ end
381
+
382
+ def watch_all_singleton_methods
383
+ fragile_methods = [:default_scope, :default_scopes, :default_scopes=] # if I touch these methods ActiveRecord breaks
384
+ watch_singleton_method *(singleton_methods - fragile_methods)
385
+ end
386
+
387
+ def watch_all_methods
388
+ watch_all_instance_methods
389
+ watch_all_singleton_methods
390
+ end
391
+
392
+ private
393
+
394
+ def add_singleton_logging method_name, options
395
+ return unless singleton_class.method_defined? method_name
396
+ m = method(method_name)
397
+ add_logging method_name, :define_singleton_method, options do |bind_object, args, &block|
398
+ m.call(*args, &block)
399
+ end
400
+ end
401
+
402
+ def add_instance_logging method_name, options
403
+ return unless method_defined? method_name
404
+ m = instance_method(method_name)
405
+ add_logging method_name, :define_method, options do |bind_object, args, &block|
406
+ m.bind(bind_object).(*args, &block)
407
+ end
408
+ end
409
+
410
+ def add_logging method_name, define_method, options, &bind_block
411
+ send(define_method, method_name) do |*args, &block|
412
+ log_args = {}
413
+ options.each do |key,value|
414
+ log_args[key] = case value
415
+ when Integer then args[value-1]
416
+ when Range then args[value]
417
+ when Symbol then eval(value.to_s)
418
+ when Proc then value.call(self)
419
+ else value
420
+ end
421
+ end
422
+ Card::Log::Performance.with_timer(method_name, log_args) do
423
+ bind_block.call(self, args, &block)
424
+ end
425
+ end
426
+ end
427
+
428
+ def log_options_variable_name method_name, define_method
429
+ "@_#{self.class.name}_#{method_name.hash.to_s.sub(/^-/,'_')}_#{define_method}_logging_options".to_sym
430
+ end
431
+
432
+ end
433
+
434
+
435
+ end
436
+
437
+ end
438
+
439
+ class Card
440
+ def self.with_logging method, opts, &block
441
+ if Card::Log::Performance.enabled_method? method
442
+ Card::Log::Performance.with_timer(method, opts) do
443
+ block.call
444
+ end
445
+ else
446
+ block.call
447
+ end
448
+ end
449
+ end
450
+