On Sat, 2007-11-17 at 20:49 +0300, Ivan Sagalaev wrote: > Hello! > > I'm about to convert my apps to play well with recently introduced > autoescaping but I have to confess that I don't get mark_safe, is_safe > and needs_autoescaping. > > First, I don't get why .is_safe attribute is needed at all. If my filter > returns any HTML I should escape it and mark_safe the result, no?
The is_safe attribute is a large time-saver when you're writing filters. Normally, you'll just want auto-escaping behaviour to be applied automatically and when writing a filter that doesn't add raw HTML markup you should be able to just write the code without having to worry about escaping. The only difficulty is when you pass a safe string into the filter. It's very easy to end up with a result that isn't a SafeData instance after a few string manipulations, so this isn't a trivial issue. For many filters, the actions they perform won't remove that safe-ness in effect, but they won't be a SafeData isntance. So Django notes that the input was a SafeData and the function is marked is_safe and, thus, it calls mark_safe() on the result so that you don't have to in your filter (all other input is automatically escaped at the right moment, since it isn't safe from further escaping). Thus, is_safe: if True, you are are guaranteeing a safe input string will always generate an output string that can be marked as safe (and Django will automatically do that for you). If not True, it is up to you to either mark the output safe manually or have it auto-escaped when auto-escaping is in effect. > Then, looking at default filters I see that .is_safe is set to False for > all filters returning non-string values. Though these values are pretty > safe for HTML when they would be converted into strings in the end. For filters returning non-strings, is_safe is a no-op, so I just picked a value. The reason False is better than True is because you don't even have to bother adding is_safe to those types of filters ("absent" defaults to False). Adding it won't harm anything, though. > And 'needs_autoescape' escapes me absolutely... If I'm dealing with user > content and HTML why, again, can't I escape it inside my filter's code > and mark_safe it? Because you wouldn't be able to write a filter that worked correctly in both auto-escaping and non-auto-escaping environments, which is a compulsory requirement in most cases. You don't want to escape inside the filter if the current context doesn't have auto-escaping in effect. The needs_autoescape attribute tells Django that your function needs to be passed a parameter called "autoescape" that is the value of the current auto-escaping effect (True or False). Yes, you can ignore needs_autoescape if you're going to restrict your filters to only working in an auto-escaping environment, but that's highly non-portable (and certainly not an option in Django's core, for example). Anybody distributing an application, for example, that was designed to work with other peoples' templates and didn't allow for auto-escaping to be either True or False at render time would have a bug in their code. > Anyway... Malcolm (as the main implementer), sorry, but the docs are > written in Linux how-to style: "make these magic passes and hope for the > best and don't try to understand the thing since you never will". A little bit hyperbolic, even for somebody who's frustrated. It is never the intention to say "you'll never understand this" and the current documentation does not even come close to saying that. When writing feedback, as welcome as it is, try to have some respect for the insane number of hours that have gone into developing this and the amount of nonsense it's generated. Consider, also, that documentation written by the person doing the design and implementation offer does sometimes miss some of the easier things because they're too easy by the time it's at a state where the documentation is written and that person is too close to things. This is just part of the ironing out problems phase. Given the types of things people complain about not understanding in the documentation (we always get requests to add things that are effectively "warning: Python will behave as it normally does and gravity has an effect on this planet."), if we go into all the fine details of how things work, the effect gets lost in the implementation. So there's a limit. Apparently you feel I've fallen short here, but it's going to be very difficult to find the middle ground. I'll have one more pass at it and after that I look forwards to reading your patch to improve things. > For example. I'm writing a filter that gets a string and wraps it's > first letter in a <b>...</b>. I'm going to split the first letter, > conditional_escape the letter and the rest, wrap a letter in <b>...</b>, > concatenate and mark_safe. Now, should I stick .is_safe? If you're always returning a safe string, then adding is_safe is a no-op. The is_safe attribute is only a necessary consideration when you aren't marking for manual safeness. > Should I also > ask for autoescape parameter and how am I supposed to use it? If your input can contain non-safe strings, you'll need to accept it. As to how to use it: test the value!! It tells you if autoescaping is in effect. From the documentation: When the filter is called, the ``autoescape`` keyword argument will be ``True`` if auto-escaping is in effect. > Ok, this was a bit messy but I honestly thought it should be easier :-) Remember that I've managed to achieve what you said was impossible in your original list of 10 things you hated about Django: auto-escaping is implemented in a way that means that almost all code is very close to backwards-compatible and filters and templates will work in both types of environments, so people don't have to write two sets of filters and two sets of templates for fragments that might be included in either environment. The effort is required by the developer rather than the template writer. Regards, Malcolm -- Why be difficult when, with a little bit of effort, you could be impossible. http://www.pointy-stick.com/blog/ --~--~---------~--~----~------------~-------~--~----~ You received this message because you are subscribed to the Google Groups "Django users" group. To post to this group, send email to django-users@googlegroups.com To unsubscribe from this group, send email to [EMAIL PROTECTED] For more options, visit this group at http://groups.google.com/group/django-users?hl=en -~----------~----~----~----~------~----~------~--~---